mmlibs/mmfw/src/Plugin/Codec/audio/MmfimaadpcmTopcm16codec.cpp
changeset 0 40261b775718
child 19 8184f75a63d4
child 24 2672ba96448e
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/mmlibs/mmfw/src/Plugin/Codec/audio/MmfimaadpcmTopcm16codec.cpp	Tue Feb 02 01:56:55 2010 +0200
@@ -0,0 +1,258 @@
+// Copyright (c) 1997-2009 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 "MMFImaAdPcmToPcm16Codec.h"
+#include "MMFImaAudioCodecState.h"
+
+#include <mmf/common/mmfaudio.h> // TMMFAudioConfig
+#include <mmf/plugin/mmfcodecimplementationuids.hrh> // KUidMmfCodecAudioSettings
+
+const TUid KUidCodecAudioConfig = {KUidMmfCodecAudioSettings};
+
+// __________________________________________________________________________
+// Implementation
+
+CMMFCodec* CMMFImaAdPcmPcm16Codec::NewL(TAny* aInitParams)
+	{
+	CMMFImaAdPcmPcm16Codec* self=new(ELeave) CMMFImaAdPcmPcm16Codec();
+	CleanupStack::PushL(self);
+	self->ConstructL(aInitParams);
+	CleanupStack::Pop(self);
+	return STATIC_CAST( CMMFCodec*, self );
+	}
+
+CMMFImaAdPcmPcm16Codec::~CMMFImaAdPcmPcm16Codec()
+	{
+	}
+
+CMMFImaAdPcmPcm16Codec::CMMFImaAdPcmPcm16Codec() : iImaAdpcmTo16Pcm(1)
+	{
+	}
+
+void CMMFImaAdPcmPcm16Codec::ConstructL(TAny*  /*aInitParams*/)
+	{
+	iTempSrcBufferPtr = iTempSrcBuffer;
+	iTempSrcBufferCount = 0;
+	
+	iChannels = 1;
+	iSamplesRate = 0;
+	iSamplesPerBlock = KImaAdpcmSamplesPerBlock;
+	iBlockAlign = KImaAdpcmBlockAlign;
+	}
+
+void CMMFImaAdPcmPcm16Codec::ResetL()
+	{
+	//Reset the actual codec
+	TMMFImaAdpcmCodecStateOld state;
+	state.iIndex = 0;
+	state.iPredicted = 0;
+	iImaAdpcmTo16Pcm.SetState(state);
+	iTempSrcBufferPtr = iTempSrcBuffer;
+	iTempSrcBufferCount = 0;
+	}
+
+/**
+CMMFImaAdPcmPcm16Codec::ProcessL
+
+This function converts IMA ADPCM samples to PCM 16 samples, 
+it is for mono ADPCM only at the moment.
+*/
+TCodecProcessResult CMMFImaAdPcmPcm16Codec::ProcessL(const CMMFBuffer& aSrc, CMMFBuffer& aDst)
+	{
+	TCodecProcessResult result;
+	result.iStatus = TCodecProcessResult::EProcessIncomplete;
+
+	//convert from generic CMMFBuffer to CMMFDataBuffer
+	iSrc = STATIC_CAST(const CMMFDataBuffer*, &aSrc);
+	iDst = STATIC_CAST(CMMFDataBuffer*, &aDst);
+
+	const TUint srcLen = iSrc->Data().Length();
+	const TUint dstMaxLen = iDst->Data().MaxLength();
+	const TUint sourceRemain = srcLen - iSrc->Position();
+
+	if (dstMaxLen < (iSamplesPerBlock * 2))
+		User::Leave(KErrArgument);
+
+	if ((iSrc->FrameNumber() == 0) && (iDst->Position() == 0))
+		{
+		ResetL();
+		}
+
+	//reset data if not a consecutive frame number
+	if ((iSrc->FrameNumber() != iLastFrameNumber) && (iSrc->FrameNumber() != (iLastFrameNumber+1)))
+		{
+		iTempSrcBufferPtr = iTempSrcBuffer;
+		iTempSrcBufferCount = 0;
+		}
+	iLastFrameNumber = iSrc->FrameNumber();
+
+	TUint dstRemain = (dstMaxLen - iDst->Position());
+	TUint srcToFillTempBuffer = 0;
+
+	//take account of src to be added to temporary buffer
+	if (iTempSrcBufferCount > 0)
+		{
+		srcToFillTempBuffer = iBlockAlign - iTempSrcBufferCount;
+		
+		if (srcToFillTempBuffer < sourceRemain) //enough source to fill temporary buffer
+			dstRemain -= (iSamplesPerBlock * 2);
+		else //not enough source to fill the temporary buffer
+			srcToFillTempBuffer = sourceRemain;
+		}
+
+	//calculate how much source is required to fill the destination buffer
+	TUint blocksRemaining = dstRemain / (iSamplesPerBlock * 2);
+	TUint maxUsableDst = blocksRemaining * iSamplesPerBlock * 2;
+	TUint srcToUse = blocksRemaining * iBlockAlign;
+
+	srcToUse += srcToFillTempBuffer;
+	srcToUse = (srcToUse < sourceRemain) ? srcToUse : sourceRemain;
+	
+	//we need to cast away CONST even on the source, as the TClass needs a TUint8*
+	TUint8* pSrc = CONST_CAST(TUint8*,iSrc->Data().Ptr());
+	pSrc += iSrc->Position();
+	TUint8* pDst = CONST_CAST(TUint8*,iDst->Data().Ptr());
+	pDst += iDst->Position();
+
+	TUint dstBytesAdded = 0;
+	TUint srcLeft = srcToUse;
+
+	//convert remaining source from previous call to ProcessL
+	if (iTempSrcBufferCount > 0)
+		{
+		//Fill temp buffer from source buffer
+		while((iTempSrcBufferCount < iBlockAlign) && (srcLeft))
+			{
+			*iTempSrcBufferPtr++ = *pSrc++;
+			iTempSrcBufferCount++;
+			srcLeft --;
+			}
+
+		if (iTempSrcBufferCount == iBlockAlign) //temp buffer full
+			{
+			//reset
+			iTempSrcBufferCount = 0;
+			iTempSrcBufferPtr = iTempSrcBuffer;
+
+			iImaAdpcmTo16Pcm.Convert(iTempSrcBufferPtr, pDst, iSamplesPerBlock);
+
+			pDst += (iSamplesPerBlock * 2);
+			dstBytesAdded += (iSamplesPerBlock * 2);
+			}
+		}
+
+	//convert full blocks
+	while (srcLeft >= iBlockAlign) 
+		{
+		iImaAdpcmTo16Pcm.Convert(pSrc, pDst, iSamplesPerBlock);
+
+		pSrc += iBlockAlign;
+		pDst += (iSamplesPerBlock * 2);
+
+		dstBytesAdded += (iSamplesPerBlock * 2);	
+		srcLeft -= iBlockAlign;	
+		}
+
+	while (srcLeft)
+		{
+		*iTempSrcBufferPtr++ = *pSrc++;
+		iTempSrcBufferCount++;
+		srcLeft--;
+		}
+	
+	//if the source buffer is consumed
+	if ((srcLen == srcToUse + iSrc->Position()))
+		{
+		if (dstBytesAdded < maxUsableDst)
+			result.iStatus = TCodecProcessResult::EDstNotFilled;
+		else
+			result.iStatus = TCodecProcessResult::EProcessComplete;
+		}
+
+	result.iSrcBytesProcessed = srcToUse;
+	result.iDstBytesAdded = dstBytesAdded;
+
+	iDst->Data().SetLength( iDst->Position() + result.iDstBytesAdded);
+
+	return result;
+	}
+
+/**
+Sets codec configuration.
+
+@param  aConfigType
+        The UID of the codec to configure.
+@param  aConfigData
+        The configuration information.
+*/
+void CMMFImaAdPcmPcm16Codec::ConfigureL(TUid aConfigType, const TDesC8& aConfigData)
+	{
+	if (aConfigType != KUidCodecAudioConfig)
+		{
+		User::Leave(KErrArgument);
+		}
+		
+	const TMMFAudioConfig& audioConfig = static_cast<const TPckgBuf<TMMFAudioConfig>&>(aConfigData)();
+	
+	iChannels = audioConfig.iChannels;
+	iSamplesRate = audioConfig.iSampleRate;
+	
+	switch (iSamplesRate * iChannels)
+		{
+		case 8000: // fall through, same as 11025
+		case 11025:
+		case 16000:
+			iBlockAlign = 256;
+			break;
+		case 22050:
+			iBlockAlign = 512;
+			break;
+			
+		case 44100:
+			iBlockAlign = 1024;
+			break;
+			
+		case 88200:
+			iBlockAlign = 2048;
+			break;
+			
+		default:
+			User::Leave(KErrArgument);
+		}
+	
+	const TUint KImaAdpcmBitsPerSample = 4;
+	// SamplesPerBlock = [(BlockAlign - 4 * Channels) * 8] / (BitsPerSample * Channels) + 1
+	iSamplesPerBlock = (iBlockAlign - 4 * iChannels) * 8 / (KImaAdpcmBitsPerSample * iChannels) + 1;
+	}
+
+TInt CMMFImaAdPcmPcm16Codec::Extension_(TUint aExtensionId, TAny*& aExtPtr, TAny*)
+	{
+	if (aExtensionId == KUidCustomInterfaceDevSoundFileBlockLength.iUid)	
+		{
+		aExtPtr = static_cast<MMMFDevSoundCustomInterfaceFileBlockLength*>(this);
+		return KErrNone;
+		}
+	else
+		{
+		return CMMFCodec::Extension_(aExtensionId, aExtPtr, NULL);	
+		}
+	}
+
+void CMMFImaAdPcmPcm16Codec::SetFileBlockLength(TUint aBlockAlign)
+	{
+	iBlockAlign = aBlockAlign;		
+	}
+