bluetooth/btstack/eirman/eirmansession.cpp
author Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
Wed, 09 Jun 2010 10:55:02 +0300
branchRCL_3
changeset 26 1f10b9300be6
parent 23 32ba20339036
permissions -rw-r--r--
Revision: 201018 Kit: 2010123

// Copyright (c) 2007-2010 Nokia Corporation and/or its subsidiary(-ies).
// All rights reserved.
// This component and the accompanying materials are made available
// under the terms of "Eclipse Public License v1.0"
// which accompanies this distribution, and is available
// at the URL "http://www.eclipse.org/legal/epl-v10.html".
//
// Initial Contributors:
// Nokia Corporation - initial contribution.
//
// Contributors:
//
// Description:
// EIR Manager server session.
// 
//

/**
 @file
 @internalComponent
*/

#include "eirmansession.h"
#include <bluetooth/logger.h>
#include <bluetooth/eirmanshared.h>
#include <bttypes.h>
#include <bluetooth/hci/hciconsts.h>
#include "eirmanpanics.h"
#include "eirmanserver.h"

#ifdef __FLOG_ACTIVE
_LIT8(KLogComponent, LOG_COMPONENT_EIRMANAGER);
#endif

CEirManSession* CEirManSession::NewL(CEirManServer& aServer, MEirSessionNotifier& aParent, TBool aInternalSession)
	{
	LOG_STATIC_FUNC
	CEirManSession* self = new(ELeave) CEirManSession(aServer, aParent, aInternalSession);
	CleanupStack::PushL(self);
	self->ConstructL();
	CleanupStack::Pop(self);
	return self;
	}

CEirManSession::CEirManSession(CEirManServer& aServer, MEirSessionNotifier& aParent, TBool aInternalSession)
	: iEirManServer(aServer)
	, iParent(aParent)
	, iEirTag(EEirTagRESERVED)
	, iInternalSession(aInternalSession)
	{
	LOG_FUNC
	}

void CEirManSession::ConstructL()
	{
	LOG_FUNC
	iEirManServer.AddSession(*this, iInternalSession);
	}

CEirManSession::~CEirManSession()
	{
	LOG_FUNC
	// deregister any registered tag that may be registered
	DeregisterTag();
	iLink.Deque();
	iEirManServer.DropSession(iInternalSession);
	}

void CEirManSession::RegisterTag(TEirTag aTag)
	{
	LOG_FUNC

	LOG1(_L("CEirManSession::RegisterTag tag = %d"), aTag);

	if(!(aTag >= EEirTagName && aTag < EEirTagRESERVED))
		{
		iParent.MesnRegisterComplete(KErrArgument);
		}
	else if(iEirTag != EEirTagRESERVED)
		{
		LOG1(_L("CEirManSession::RegisterTag ERROR, Tag in use: %d"), iEirTag);
		iParent.MesnRegisterComplete(KErrInUse);
		}
	else
		{
		// Register this tag for callbacks
		if(iEirManServer.EirFeatureState() == EEirFeatureReady)
			{
			// Eir is supported
			TInt result = iEirManServer.EirManager().RegisterTag(aTag, *this);
			if(result == KErrNone)
				{
				iEirTag = aTag;
				}
			iParent.MesnRegisterComplete(result);
			}
		else if(iEirManServer.EirFeatureState() == EEirFeaturePending)
			{
			// We don't know if eir is supported or not at this moment
			iRegisterPending = ETrue;
			iPendingEirTag = aTag;
			}
		else
			{
			// Eir is not supported
			iParent.MesnRegisterComplete(KErrNotSupported);
			}
		}
	}

void CEirManSession::DeregisterTag()
	{
	LOG_FUNC
	LOG1(_L("CEirManSession::DeregisterTag tag = %d"), iEirTag);
	
	if(iEirTag != EEirTagRESERVED)
		{
		// Deregister this tag for callbacks 
		iEirManServer.EirManager().DeregisterTag(iEirTag);
		iEirTag = EEirTagRESERVED;
		}
	}

// Eir Server has tried to register tag
void CEirManSession::NotifyEirFeatureState(TInt aResult)
	{
	LOG1(_L("Eir Server has been notified feature ready, result: %d"), aResult);
	if(aResult == KErrNone && iRegisterPending)
		{
		__ASSERT_DEBUG(iEirManServer.EirFeatureState() == EEirFeatureReady, EIR_SESSION_PANIC(EEirSessionEirFeatureNotSupported));
		TInt result = iEirManServer.EirManager().RegisterTag(iPendingEirTag, *this);
		if(result == KErrNone)
			{
			iEirTag = iPendingEirTag;
			}
		iRegisterPending = EFalse;
		iParent.MesnRegisterComplete(result);
		}
	else if(iRegisterPending)
		{
		iRegisterPending = EFalse;
		iParent.MesnRegisterComplete(aResult);
		}
	}


TInt CEirManSession::NewData(TInt aRequiredLength)
	{
	LOG_FUNC
	return iEirManServer.EirManager().NewData(iEirTag, aRequiredLength);
	}


TInt CEirManSession::SetData(const TDesC8& aData, TEirDataMode aMode)
	{
	LOG_FUNC
	return iEirManServer.EirManager().SetData(iEirTag, aData, aMode);
	}

// Callback from the EIR Manager
void CEirManSession::MemnEirBlockAvailable(TEirTag aTag, TUint aSpaceForTag)
	{
	LOG_FUNC
	LOG2(_L("CEirManSession::MemnEirBlockAvailable Tag: %d space: %d"), aTag, aSpaceForTag);

	// Silently discard callbacks not containing our EIR Tag
	if(aTag != iEirTag)
		{
		__ASSERT_DEBUG(EFalse, EIR_SESSION_PANIC(EEirSessionInvalidEirTag));
		LOG3(_L("CEirManSession::MemnEirBlockAvailable early return: aTag: %d iEirTag: %d space: %d"), aTag, iEirTag, aSpaceForTag);
		return;
		}

	iParent.MesnSpaceAvailable(aSpaceForTag);

	}

TEirTag CEirManSession::EirTag() const
	{
	return iEirTag;
	}

CEirManExternalSession* CEirManExternalSession::NewL(CEirManServer& aServer)
	{
	CEirManExternalSession* self = new(ELeave) CEirManExternalSession();
	CleanupStack::PushL(self);
	self->ConstructL(aServer);
	CleanupStack::Pop(self);
	return self;
	}

CEirManExternalSession::CEirManExternalSession()
	{
	
	}

void CEirManExternalSession::ConstructL(CEirManServer& aServer)
	{
	iSession = CEirManSession::NewL(aServer, *this, EFalse);
	}

CEirManExternalSession::~CEirManExternalSession()
	{
	if(!iDataAvailableListenerMessage.IsNull())
		{
		// complete any outstanding messages.
		iDataAvailableListenerMessage.Complete(KErrCancel);
		}
	delete iSession;
	}
/*
 * Each of the individual methods is responsible for completing the message.
 * All 'leaves' are processed in CSession2::ServiceError() resulting in message completion
 * with the appropriate error code.
 * RegisterTagL() and RegisterSpaceAvailableListenerL() store aMessage parameter for the future use
 * when the callbacks are called.
 */
void CEirManExternalSession::ServiceL(const RMessage2& aMessage)
	{
	LOG_FUNC
	LOG1(_L("CEirManSession::ServiceL aMessage.Function() = %d"), aMessage.Function());

	switch (aMessage.Function())
		{
	case EEirManRegisterTag:
		RegisterTagL(aMessage);
		break;

	case EEirManSpaceAvailableNotification:
		RegisterSpaceAvailableListenerL(aMessage);
		break;

	case EEirManCancelSpaceAvailableNotification:
		CancelSpaceAvailableListenerL(aMessage);
		break;

	case EEirManSetData:
		SetDataL(aMessage);
		break;
		
	case EEirManNewData:
		NewDataL(aMessage);
		break;

	default:
		aMessage.Panic(KEirManCliPncCat, EEirManPanicInvalidIPC);
		break;
		}
	}

void CEirManExternalSession::RegisterTagL(const RMessage2& aMessage)
	{
	LOG_FUNC
	TEirTag tag = static_cast<TEirTag>(aMessage.Int0());
	LOG1(_L("CEirManSession::RegisterTag tag = %d"), tag);

	iRegisterMessage = aMessage;
	
	iSession->RegisterTag(tag);
	}

void CEirManExternalSession::MesnRegisterComplete(TInt aResult)
	{
	if (aResult == KErrArgument)
		{
		iRegisterMessage.Panic(KEirManCliPncCat, EEirManPanicInvalidTag);
		}
	else
		{
		iRegisterMessage.Complete(aResult);
		}
	}

void CEirManExternalSession::RegisterSpaceAvailableListenerL(const RMessage2& aMessage)
	{
	LOG_FUNC

	if(iDataAvailableListenerMessage.Handle())
		{
		LOG(_L("CEirManSession:::RegisterSpaceAvailableListener ERROR IN USE"));
		LEAVEL(KErrInUse);
		}

	iDataAvailableListenerMessage = aMessage;

	if(iLastSpaceOffered != 0)
		{
		LOG(_L("cached space present, completing immediately"));
		CompleteSpaceAvailableRequest(iLastSpaceOffered);
		iLastSpaceOffered = 0;
		return;
		}
	
	LOG(_L("waiting for callback..."));
	}
	
void CEirManExternalSession::NewDataL(const RMessage2& aMessage)
	{
	LOG_FUNC

	if (iSession->EirTag() == EEirTagRESERVED)
		{
		aMessage.Panic(KEirManCliPncCat, EEirManPanicInvalidTag);
		return;
		}

	TInt requiredLength = static_cast<TInt>(aMessage.Int0());

	LEAVEIFERRORL(iSession->NewData(requiredLength));
	aMessage.Complete(KErrNone);
	}

void CEirManExternalSession::CancelSpaceAvailableListenerL(const RMessage2& aMessage)
	{
	LOG_FUNC

	if(!iDataAvailableListenerMessage.Handle())
		{
		LEAVEL(KErrNotFound);
		}

	iDataAvailableListenerMessage.Complete(KErrCancel);
	aMessage.Complete(KErrNone);
	}

void CEirManExternalSession::SetDataL(const RMessage2& aMessage)
	{
	LOG_FUNC
	if (iSession->EirTag() == EEirTagRESERVED)
		{
		aMessage.Panic(KEirManCliPncCat, EEirManPanicInvalidTag);
		return;
		}

	TEirDataMode eirDataMode = static_cast<TEirDataMode>(aMessage.Int1());
	LOG1(_L("EirDataMode: %d"),  eirDataMode);

	// No need to allocate memory with an expensive malloc() call (via HBuf8::NewL or whatever), 
	// since the EIR contents fit in the stack, and the EIR Manager will cache the data anyway
	TBuf8<KHCIExtendedInquiryResponseMaxLength> data;
	LEAVEIFERRORL(aMessage.Read(0, data));
	
	LEAVEIFERRORL(iSession->SetData(data, eirDataMode));
	aMessage.Complete(KErrNone);
	}

void CEirManExternalSession::MesnSpaceAvailable(TUint aSpaceForTag)
	{
	if(iDataAvailableListenerMessage.Handle())
		{
		LOG(_L("Listener outstanding, completing request"));
		// Clear stored value
		iLastSpaceOffered = 0;
		CompleteSpaceAvailableRequest(aSpaceForTag);
		}
	else
		{
		LOG(_L("NO Listener outstanding, storing space"));
		// Store in case a client registers later. If this happens multiple times, only the last one will be remembered
		iLastSpaceOffered = aSpaceForTag;
		}
	}

void CEirManExternalSession::CompleteSpaceAvailableRequest(TUint aBytesAvailable)
	{
	LOG_FUNC
	LOG1(_L("CEirManSession::CompleteSpaceAvailableRequest bytes: %d"), aBytesAvailable);
	// Offer the space to the client
	TPckgC<TUint32> pckg(aBytesAvailable);

	TInt err = iDataAvailableListenerMessage.Write(0, pckg);
	iDataAvailableListenerMessage.Complete(err);
	}

CEirManInternalSession* CEirManInternalSession::NewL(CEirManServer& aServer, MEirInternalSessionNotifier& aParent)
	{
	LOG_STATIC_FUNC
	CEirManInternalSession* self = new(ELeave) CEirManInternalSession(aParent);
	CleanupStack::PushL(self);
	self->ConstructL(aServer);
	CleanupStack::Pop(self);
	return self;
	}

CEirManInternalSession::CEirManInternalSession(MEirInternalSessionNotifier& aParent)
	: iParent(aParent)
	{
	
	}

void CEirManInternalSession::ConstructL(CEirManServer& aServer)
	{
	iSession = CEirManSession::NewL(aServer, *this, ETrue);
	iSetDataCb = new (ELeave) CAsyncCallBack(CActive::EPriorityHigh);
	TCallBack cb(&SetDataCb, this);
	iSetDataCb->Set(cb);
	}

CEirManInternalSession::~CEirManInternalSession()
	{
	delete iSetDataCb;
	delete iSession;
	}
	
void CEirManInternalSession::RegisterTag(TEirTag aTag)
	{
	iSession->RegisterTag(aTag);
	}

void CEirManInternalSession::SetData(const TDesC8& aData, TEirDataMode aMode)
	{
	delete iPendingData;
	iPendingData = NULL;
	iSetDataCb->Cancel();
	iPendingData = aData.Alloc();
	iPendingMode = aMode;
	if (iPendingData)
		{
		iSetDataCb->CallBack();
		}
	}

TInt CEirManInternalSession::SetDataCb(TAny* aThis)
	{
	static_cast<CEirManInternalSession*>(aThis)->DoSetData();
	return KErrNone;
	}

void CEirManInternalSession::DoSetData()
	{
	TInt err = iSession->SetData(*iPendingData, iPendingMode);
	if (err != KErrNone)
		{
		iParent.MeisnSetDataError(err);
		}
	}

TInt CEirManInternalSession::NewData(TInt aRequiredLength)
	{
	return iSession->NewData(aRequiredLength);
	}


void CEirManInternalSession::MesnRegisterComplete(TInt aResult)
	{
	iParent.MeisnRegisterComplete(aResult);
	}

void CEirManInternalSession::MesnSpaceAvailable(TUint aSpaceForTag)
	{
	iParent.MeisnSpaceAvailable(aSpaceForTag - KEirLengthTagLength);
	}