devsound/devsoundpluginsupport/src/CustomInterfaces/cmmfdevsoundcibitrate.cpp
author hgs
Fri, 08 Oct 2010 19:40:43 +0100
changeset 0 79dd3e2336a0
permissions -rw-r--r--
2010wk36_01

/*
* Copyright (c) 2005 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:
*
*/


#include "cmmfdevsoundcibitrateimplementationuid.hrh"

#include <ecom/implementationproxy.h>
#include <ecom/ecom.h>
#include <s32mem.h>

#include "cmmfdevsoundcibitrate.h"

// __________________________________________________________________________
// Implementation

////////////////////////////////////////// MUX /////////////////////

TInt CMMFDevSoundCIBitRateMux::OpenInterface(TUid /*aInterfaceId*/)
	{
	// attempt to open the interface link with the
	// remote slave device
	iRemoteHandle = -1;
	TUid slaveId = {KMmfUidCustomInterfaceBitRateDeMux};
		
	TInt handle = iUtility->OpenSlave(slaveId, KNullDesC8);
	if (handle >= 0)
		{
		iRemoteHandle = handle;
		}
		
	return iRemoteHandle;
	}

void CMMFDevSoundCIBitRateMux::Release()
	{
	// close the slave device if it exists
	if (iRemoteHandle != -1)
		{
		// we assume the slave is closed correctly
		iUtility->CloseSlave(iRemoteHandle);
		}
	
	TUid key = iKey;
	delete this;
	
	// tell ECom to destroy us
	REComSession::DestroyedImplementation(key);
	}
	
void CMMFDevSoundCIBitRateMux::PassDestructorKey(TUid aDestructorKey)
	{
	// store the destructor key
	iKey = aDestructorKey;
	}

void CMMFDevSoundCIBitRateMux::CompleteConstructL(MMMFDevSoundCustomInterfaceMuxUtility* aCustomUtility)
	{
	// store a pointer to the utility
	iUtility = aCustomUtility;
	}
	
MMMFDevSoundCustomInterfaceMuxPlugin* CMMFDevSoundCIBitRateMux::NewL()
	{
	CMMFDevSoundCIBitRateMux* self = new (ELeave) CMMFDevSoundCIBitRateMux;
	return self;
	}
	
TAny* CMMFDevSoundCIBitRateMux::CustomInterface(TUid /*aInterfaceId*/)
	{
	MMMFDevSoundCustomInterfaceBitRate* interface = this;
	return interface;
	}
	
CMMFDevSoundCIBitRateMux::CMMFDevSoundCIBitRateMux() :
	iRemoteHandle(-1)
	{	
	}

CMMFDevSoundCIBitRateMux::~CMMFDevSoundCIBitRateMux()
	{	
	}

// from MMMFDevSoundCustomInterfaceBitRate
void CMMFDevSoundCIBitRateMux::GetSupportedBitRatesL(RArray<TInt>& aSupportedBitRates)
	{
	if (iRemoteHandle == -1)
		{
		User::Leave(KErrNotReady);
		}
	
	// first clear out the array
	aSupportedBitRates.Reset();
	
	// now fetch the count from the server
	TInt count = -1;
	count = iUtility->SendSlaveSyncCommand(iRemoteHandle, EMMFDevSoundCIBitRateGetSupportedBitRates, KNullDesC8);
	
	// if count is negative then the server side left with an error
	if (count < 0)
		{
		User::Leave(count);
		}
	
	// no point getting the data if the count is zero
	if (count != 0)
		{
		// allocate a temporary buffer to hold the bitrates
		HBufC8* buf = HBufC8::NewLC(count * sizeof(TInt32));
		TPtr8 ptr = buf->Des();
	
		// fetch the bitrates - but send over the received count to be sure
		TPckgBuf<TInt> countBuf(count);
		User::LeaveIfError(iUtility->SendSlaveSyncCommandResult(
											 iRemoteHandle, 
											 EMMFDevSoundCIBitRateGetSupportedBitRatesArray,
											 countBuf, ptr));
	
		// stream data into the pointer
		RDesReadStream stream(ptr);
		CleanupClosePushL(stream);
				
		TInt err = KErrNone;
		for (TInt i = 0; i < count; i++)
			{
			// note we don't destroy array because we don't own it
			// but we do reset it as it is incomplete
			err = aSupportedBitRates.Append(stream.ReadInt32L());
			if (err != KErrNone)
				{
				aSupportedBitRates.Reset();
				User::Leave(KErrCorrupt);
				}
			}
		
		CleanupStack::PopAndDestroy(2, buf);// stream, buf
		}
	}
	
TInt CMMFDevSoundCIBitRateMux::BitRateL()
	{
	if (iRemoteHandle == -1)
		{
		User::Leave(KErrNotReady);
		}
	
	// send EMMFDevSoundCIBitRateBitRate command to slave
	TInt bitrate = 0;
	bitrate = iUtility->SendSlaveSyncCommand(iRemoteHandle, EMMFDevSoundCIBitRateBitRate, KNullDesC8);
	
	// if bitrate is negative then remote side left with an error
	if (bitrate < 0)
		{
		User::Leave(bitrate);
		}
	
	return bitrate;
	}
	
void CMMFDevSoundCIBitRateMux::SetBitRateL(TInt aBitRate)
	{
	if (iRemoteHandle == -1)
		{
		User::Leave(KErrNotReady);
		}
	
	// send the bitrate in the sync command
	TPckgBuf<TInt> bitBuffer(aBitRate);
	
	// any return code other than zero is an error
	User::LeaveIfError(iUtility->SendSlaveSyncCommand(iRemoteHandle, EMMFDevSoundCIBitRateSetBitRate, bitBuffer));
	}

/////////////////////////////////////// DEMUX //////////////////////	
	

TInt CMMFDevSoundCIBitRateDeMux::OpenInterface(TUid /*aInterfaceId*/)
	{
	return KErrNone;
	}
	
void CMMFDevSoundCIBitRateDeMux::Release()
	{
	TUid key = iKey;
	
	delete this;
	
	// tell ECom to destroy us
	REComSession::DestroyedImplementation(key);
	}
	
void CMMFDevSoundCIBitRateDeMux::PassDestructorKey(TUid aDestructorKey)
	{
	// store the destructor key
	iKey = aDestructorKey;
	}
	
void CMMFDevSoundCIBitRateDeMux::SetInterfaceTarget(MMMFDevSoundCustomInterfaceTarget* aTarget)
	{
	iTarget = aTarget;
	}
	
void CMMFDevSoundCIBitRateDeMux::CompleteConstructL(MMMFDevSoundCustomInterfaceDeMuxUtility* aCustomUtility)
	{
	// store a pointer to the utility
	iUtility = aCustomUtility;
	}

void CMMFDevSoundCIBitRateDeMux::RefreshL()
	{
	// refetch the bitrate custom interface if we already have a target
	if (iTarget)
		{
		MMMFDevSoundCustomInterfaceBitRate* ptr = NULL;
		ptr = static_cast <MMMFDevSoundCustomInterfaceBitRate*> (iTarget->CustomInterface(KUidCustomInterfaceDevSoundBitRate));
	
		if (!ptr)
			{
			iBitRateInterface = NULL;
			User::Leave(KErrNotSupported);
			}
		else
			{
			iBitRateInterface = ptr;
			}	
		}
	}


MMMFDevSoundCustomInterfaceDeMuxPlugin* CMMFDevSoundCIBitRateDeMux::NewL()
	{
	CMMFDevSoundCIBitRateDeMux* self = new (ELeave) CMMFDevSoundCIBitRateDeMux;
	return self;
	}
	
CMMFDevSoundCIBitRateDeMux::CMMFDevSoundCIBitRateDeMux()
	{
	
	}

CMMFDevSoundCIBitRateDeMux::~CMMFDevSoundCIBitRateDeMux()
	{
	iBitRateArray.Reset();
	iBitRateArray.Close();
	}


TInt CMMFDevSoundCIBitRateDeMux::DoOpenSlaveL(TUid /*aInterface*/, const TDesC8& /*aPackageBuf*/)
	{
	// fetch the bitrate custom interface
	MMMFDevSoundCustomInterfaceBitRate* ptr = NULL;
	ptr = static_cast <MMMFDevSoundCustomInterfaceBitRate*> (iTarget->CustomInterface(KUidCustomInterfaceDevSoundBitRate));
	
	if (!ptr)
		{
		iBitRateInterface = NULL;
		User::Leave(KErrNotSupported);
		}
	else
		{
		iBitRateInterface = ptr;
		}
	return KErrNone;
	}
	
void CMMFDevSoundCIBitRateDeMux::DoCloseSlaveL(TInt /*aHandle*/)
	{
	// nothing to do
	}

// original RMessage is supplied so that remote demux plugin can extract necessary details
// using DeMux utility
TInt CMMFDevSoundCIBitRateDeMux::DoSendSlaveSyncCommandL(const RMmfIpcMessage& aMessage)
	{
	TMMFDevSoundCIMessageData data;
	
	// decode message
	iUtility->GetSyncMessageDataL(aMessage, data);
	TInt retVal = -1;
	
	switch (data.iCommand)
		{
		case EMMFDevSoundCIBitRateBitRate:
			{
			retVal = DoBitRateL();
			break;
			}
		case EMMFDevSoundCIBitRateSetBitRate:
			{
			// we know that offset 2 contains a TInt
			TPckgBuf<TInt> bitBuffer;
			iUtility->ReadFromInputDesL(aMessage, &bitBuffer);
			
			DoSetBitRateL(bitBuffer());
			retVal = KErrNone;
			break;
			}
		case EMMFDevSoundCIBitRateGetSupportedBitRates:
			{
			// reset the current bitrate array
			iBitRateArray.Reset();
			DoGetSupportedBitRatesL(iBitRateArray);
			
			// send back the array count
			TInt count = iBitRateArray.Count();
			retVal = count;
			break;
			}
		default:
			{
			User::Leave(KErrNotSupported);
			}
		}
		
	return retVal;
	}
	
TInt CMMFDevSoundCIBitRateDeMux::DoSendSlaveSyncCommandResultL(const RMmfIpcMessage& aMessage)
	{
	TMMFDevSoundCIMessageData data;
	
	// decode message
	iUtility->GetSyncMessageDataL(aMessage, data);
	
	switch (data.iCommand)
		{
		case EMMFDevSoundCIBitRateGetSupportedBitRatesArray:
			{
			DoCopyBitRateBufferToClientL(aMessage);		
			break;
			}
		default:
			{
			User::Leave(KErrNotSupported);
			}
		}
	return KErrNone;
	}
	
void CMMFDevSoundCIBitRateDeMux::DoSendSlaveAsyncCommandL(const RMmfIpcMessage& /*aMessage*/)
	{
	// not used in this interface
	}
	
void CMMFDevSoundCIBitRateDeMux::DoSendSlaveAsyncCommandResultL(const RMmfIpcMessage& /*aMessage*/)
	{
	// not used in this interface
	}

// bitrate custom interface implementation
void CMMFDevSoundCIBitRateDeMux::DoGetSupportedBitRatesL(RArray<TInt>& aSupportedBitRates)
	{
	if (!iBitRateInterface)
		{
		User::Leave(KErrNotReady);
		}
		
	iBitRateInterface->GetSupportedBitRatesL(aSupportedBitRates);
	}


void CMMFDevSoundCIBitRateDeMux::DoCopyBitRateBufferToClientL(const RMmfIpcMessage& aMessage)
	{
	if (!iBitRateInterface)
		{
		User::Leave(KErrNotReady);
		}
		
	// check our count is the same as the client's
	TPckgBuf<TInt> countBuffer;
	iUtility->ReadFromInputDesL(aMessage, &countBuffer);

	TInt count = countBuffer();
	if (count != iBitRateArray.Count())
		{
		User::Leave(KErrCorrupt);
		}
			
	// send back the array - the client has the count already
	const TInt KBufExpandSize8 = 8; //two TInt's
	CBufFlat* dataCopyBuffer = CBufFlat::NewL(KBufExpandSize8);
	CleanupStack::PushL(dataCopyBuffer);
	RBufWriteStream stream;
	stream.Open(*dataCopyBuffer);
	CleanupClosePushL(stream);
	
	for (TInt i = 0; i < count; i++)
		{
		stream.WriteInt32L(iBitRateArray[i]);
		}
	
	// write the data to the supplied descriptor buffer
	TPtr8 ptrBuf = dataCopyBuffer->Ptr(0);
	iUtility->WriteToOutputDesL(aMessage, ptrBuf);
	stream.Close();

	CleanupStack::PopAndDestroy(2); // iDataCopyBuffer, stream
	}


TInt CMMFDevSoundCIBitRateDeMux::DoBitRateL()
	{
	if (!iBitRateInterface)
		{
		User::Leave(KErrNotReady);
		}
	return iBitRateInterface->BitRateL();
	}

void CMMFDevSoundCIBitRateDeMux::DoSetBitRateL(TInt aBitRate)
	{
	if (!iBitRateInterface)
		{
		User::Leave(KErrNotReady);
		}
			
	// set the bitrate		
	iBitRateInterface->SetBitRateL(aBitRate);
	}



//
// ImplementationTable
//

const TImplementationProxy ImplementationTable[] = 
	{
	IMPLEMENTATION_PROXY_ENTRY(KMmfUidCustomInterfaceBitRateMux,	CMMFDevSoundCIBitRateMux::NewL),
	IMPLEMENTATION_PROXY_ENTRY(KMmfUidCustomInterfaceBitRateDeMux,	CMMFDevSoundCIBitRateDeMux::NewL),
	};


//
// ImplementationGroupProxy
//
////////////////////////////////////////////////////////////////////////////////

EXPORT_C const TImplementationProxy* ImplementationGroupProxy(TInt& aTableCount)
	{
	aTableCount = sizeof(ImplementationTable) / sizeof(TImplementationProxy);

	return ImplementationTable;
	}