Update contrib.
1 // Copyright (c) 2003-2009 Nokia Corporation and/or its subsidiary(-ies).
2 // All rights reserved.
3 // This component and the accompanying materials are made available
4 // under the terms of "Eclipse Public License v1.0"
5 // which accompanies this distribution, and is available
6 // at the URL "http://www.eclipse.org/legal/epl-v10.html".
8 // Initial Contributors:
9 // Nokia Corporation - initial contribution.
16 #include "MMFAudioCodec.h"
17 #include <mmf/common/mmfpaniccodes.h>
18 #include "MmfImaAdpcmtopcm16hwdevice.h"
19 #include <mmf/server/mmfswcodecwrappercustominterfacesuids.hrh>
26 CMMFImaAdpcmToPcm16CodecHwDevice* CMMFImaAdpcmToPcm16CodecHwDevice::NewL()
28 CMMFImaAdpcmToPcm16CodecHwDevice* self=new(ELeave)CMMFImaAdpcmToPcm16CodecHwDevice();
29 CleanupStack::PushL(self);
31 CleanupStack::Pop(self);
40 void CMMFImaAdpcmToPcm16CodecHwDevice::ConstructL()
42 iCodec = new (ELeave) CMMFImaAdpcmToPcm16Codec();
47 * ~CMMFMulawPcm16HwDevice
50 CMMFImaAdpcmToPcm16CodecHwDevice::~CMMFImaAdpcmToPcm16CodecHwDevice()
59 CMMFSwCodec &CMMFImaAdpcmToPcm16CodecHwDevice::Codec()
65 Retrieves a custom interface to the device.
66 The reference CMMFImaAdpcmToPcm16CodecHwDevice supports one FileBlockLength custom interface,
67 or else calls two standard (CMMFSwCodecWrapper) custom interfaces,TPlayCustomInterface and TRecordCustomInterface.
70 Interface UID, defined with the custom interface.
71 aInterface = KUidCustomInterfaceDevSoundFileBlockLength,
72 aInterface = KMmfPlayCustomInterface for TPlayCustomInterface,
73 aInterface = KMmfRecordCustomInterface for TRecordCustomInterface.
74 @return A pointer to the interface implementation, The return value must be cast to the
75 correct type by the user.
77 TAny* CMMFImaAdpcmToPcm16CodecHwDevice::CustomInterface(TUid aInterfaceId)
79 if(aInterfaceId == KUidCustomInterfaceDevSoundFileBlockLength)
81 return static_cast<MMMFDevSoundCustomInterfaceFileBlockLength*>(this);
85 return CMMFSwCodecWrapper::CustomInterface(aInterfaceId);
90 @see CMMFSwCodecWrapper::Start
92 this function sets SampleRate and Channels for CMMFImaAdpcmToPcm16Codec
94 TInt CMMFImaAdpcmToPcm16CodecHwDevice::Start(TDeviceFunc aFuncCmd, TDeviceFlow aFlowCmd)
96 TInt err = static_cast<CMMFImaAdpcmToPcm16Codec*>(iCodec)->Configure(iChannels, iSampleRate, iBlockAlign);
99 err = CMMFSwCodecWrapper::Start(aFuncCmd, aFlowCmd);
108 This function sets file's block length for CMMFImaAdpcmToPcm16CodecHwDevice
111 The file's block length
113 void CMMFImaAdpcmToPcm16CodecHwDevice::SetFileBlockLength(TUint aBlockAlign)
115 iBlockAlign = aBlockAlign;
118 CMMFImaAdpcmToPcm16Codec::CMMFImaAdpcmToPcm16Codec()
122 iBlockAlign = KImaAdpcmBlockAlign;
123 iSamplesPerBlock = KImaAdpcmSamplesPerBlock;
130 * @pre position of buffer aSrc is 0
131 * @pre position of buffer aDst is 0
132 * @pre sufficient bytes in output to consume input
133 * @return TCodecProcessResult
134 * This function converts IMA ADPCM samples to PCM samples.
137 CMMFSwCodec::TCodecProcessResult CMMFImaAdpcmToPcm16Codec::ProcessL(const CMMFBuffer& aSrc, CMMFBuffer& aDst)
139 CMMFSwCodec::TCodecProcessResult result;
140 result.iCodecProcessStatus = TCodecProcessResult::EProcessComplete;
142 //convert from generic CMMFBuffer to CMMFDataBuffer
143 const CMMFDataBuffer* src = STATIC_CAST(const CMMFDataBuffer*, &aSrc);
144 CMMFDataBuffer* dst = STATIC_CAST(CMMFDataBuffer*, &aDst);
146 if( !CheckPreconditions( src, dst ) )
148 //[ precondition(s) violation ]
149 User::Leave(KErrArgument);
152 //calculate how much source is required to fill the destination buffer
153 TUint blocksRemaining = src->Data().Length() / iBlockAlign;
155 //we need to cast away CONST even on the source, as the TClass needs a TUint8*
156 TUint8* pSrc = CONST_CAST(TUint8*,src->Data().Ptr());
157 TUint8* pDst = CONST_CAST(TUint8*,dst->Data().Ptr());
159 //[ [process full blocks ]
160 TUint dstBytesAdded = 0;
161 for( TUint count = 0; count < blocksRemaining; count++ )
163 iImaAdpcmTo16Pcm.Convert(pSrc, pDst, iSamplesPerBlock);
165 pDst += (iSamplesPerBlock * sizeof(TInt16));
166 dstBytesAdded += (iSamplesPerBlock * sizeof(TInt16));
169 result.iCodecProcessStatus = TCodecProcessResult::EProcessComplete;
170 result.iSrcBytesProcessed = blocksRemaining * iBlockAlign;
171 result.iDstBytesAdded = dstBytesAdded;
172 dst->Data().SetLength(result.iDstBytesAdded);
174 //[ check post conditions
175 __ASSERT_DEBUG( (src->Position() == 0), TMmfAudioCodecPanicsNameSpace::Panic( TMmfAudioCodecPanicsNameSpace::EPostConditionViolation ));
176 __ASSERT_DEBUG( (dst->Position() == 0), TMmfAudioCodecPanicsNameSpace::Panic( TMmfAudioCodecPanicsNameSpace::EPostConditionViolation ));
177 TInt r1 = src->Data().Length();
179 TInt r2 = dst->Data().Length();
180 r2 /=(iSamplesPerBlock * sizeof(TInt16));
181 __ASSERT_DEBUG( r1== r2, TMmfAudioCodecPanicsNameSpace::Panic(TMmfAudioCodecPanicsNameSpace::EPostConditionViolation ));
182 __ASSERT_DEBUG( dst->Data().Length() % 2 == 0, TMmfAudioCodecPanicsNameSpace::Panic( TMmfAudioCodecPanicsNameSpace::EPostConditionViolation )); // pcm output
190 * This methos tests the preconditions of the ProcessL method
191 * @return TBool ETrue for sucess and EFalse for failure of the preconditions
194 TBool CMMFImaAdpcmToPcm16Codec::CheckPreconditions( const CMMFDataBuffer* aSrcBuffer, CMMFDataBuffer* aDestBuffer )
196 TBool result = EFalse;
208 // Check position of src and dest are 0
209 if( aSrcBuffer->Position() )
214 // Check position of src and dest are 0
215 if( aDestBuffer->Position() )
220 // check there are sufficient bytes in the output to consume the input
221 const TUint KTempBufferSize = iSamplesPerBlock * 2;
222 TInt numInputSubFrames = aSrcBuffer->Data().Length() / iBlockAlign;
223 TInt numOutputSubFrames = aDestBuffer->Data().MaxLength() / KTempBufferSize;
225 //[ we need modulo 1010 bytes on all src frames that are not the last
227 // For the last frame we will code only whole frames and effectively
228 // drop any remaining samples]
229 TBool validInputDataLength = (aSrcBuffer->Data().Length() % iBlockAlign == 0) ;
231 if( (numInputSubFrames > numOutputSubFrames) || // sufficient space in the output for the input
232 (aSrcBuffer->Position() > 0 ) || // position must be zero since we can eat all the data
233 (aDestBuffer->Position() > 0 ) ||
234 (!validInputDataLength)) //position must be zero
239 result = ETrue; // preconditions have been satisfied
247 This function sets file's block length, channels & sample rate.
250 The file's number of channels
252 The file's sample rate
254 The file's block length
256 TInt CMMFImaAdpcmToPcm16Codec::Configure(TUint aChannels, TUint aSampleRate, TUint aBlockAlign)
258 iChannels = aChannels;
259 iSampleRate = aSampleRate;
261 if (aBlockAlign < 256)
263 switch (iSampleRate * iChannels)
265 case 8000: // fall through, same as 11025
288 iBlockAlign = aBlockAlign;
291 const TUint KImaAdpcmBitsPerSample = 4;
292 iSamplesPerBlock = (iBlockAlign - 4 * iChannels) * 8 / (KImaAdpcmBitsPerSample * iChannels) + 1;