1.1 --- /dev/null Thu Jan 01 00:00:00 1970 +0000
1.2 +++ b/os/mm/devsound/devsoundrefplugin/src/plugin/audio/Mmfimaadpcmtopcm16hwdevice.cpp Fri Jun 15 03:10:57 2012 +0200
1.3 @@ -0,0 +1,301 @@
1.4 +// Copyright (c) 2003-2009 Nokia Corporation and/or its subsidiary(-ies).
1.5 +// All rights reserved.
1.6 +// This component and the accompanying materials are made available
1.7 +// under the terms of "Eclipse Public License v1.0"
1.8 +// which accompanies this distribution, and is available
1.9 +// at the URL "http://www.eclipse.org/legal/epl-v10.html".
1.10 +//
1.11 +// Initial Contributors:
1.12 +// Nokia Corporation - initial contribution.
1.13 +//
1.14 +// Contributors:
1.15 +//
1.16 +// Description:
1.17 +//
1.18 +
1.19 +#include "MMFAudioCodec.h"
1.20 +#include <mmf/common/mmfpaniccodes.h>
1.21 +#include "MmfImaAdpcmtopcm16hwdevice.h"
1.22 +#include <mmf/server/mmfswcodecwrappercustominterfacesuids.hrh>
1.23 +
1.24 +/**
1.25 +*
1.26 +* NewL
1.27 +*
1.28 +*/
1.29 +CMMFImaAdpcmToPcm16CodecHwDevice* CMMFImaAdpcmToPcm16CodecHwDevice::NewL()
1.30 + {
1.31 + CMMFImaAdpcmToPcm16CodecHwDevice* self=new(ELeave)CMMFImaAdpcmToPcm16CodecHwDevice();
1.32 + CleanupStack::PushL(self);
1.33 + self->ConstructL();
1.34 + CleanupStack::Pop(self);
1.35 + return self;
1.36 + }
1.37 +
1.38 +/**
1.39 +*
1.40 +* ConstructL
1.41 +*
1.42 +*/
1.43 +void CMMFImaAdpcmToPcm16CodecHwDevice::ConstructL()
1.44 + {
1.45 + iCodec = new (ELeave) CMMFImaAdpcmToPcm16Codec();
1.46 + }
1.47 +
1.48 +/**
1.49 +*
1.50 +* ~CMMFMulawPcm16HwDevice
1.51 +*
1.52 +*/
1.53 +CMMFImaAdpcmToPcm16CodecHwDevice::~CMMFImaAdpcmToPcm16CodecHwDevice()
1.54 + {
1.55 + }
1.56 +
1.57 +/**
1.58 +*
1.59 +* Codec
1.60 +*
1.61 +*/
1.62 +CMMFSwCodec &CMMFImaAdpcmToPcm16CodecHwDevice::Codec()
1.63 + {
1.64 + return *iCodec;
1.65 + }
1.66 +
1.67 +/**
1.68 +Retrieves a custom interface to the device.
1.69 +The reference CMMFImaAdpcmToPcm16CodecHwDevice supports one FileBlockLength custom interface,
1.70 +or else calls two standard (CMMFSwCodecWrapper) custom interfaces,TPlayCustomInterface and TRecordCustomInterface.
1.71 +
1.72 +@param aInterface
1.73 + Interface UID, defined with the custom interface.
1.74 + aInterface = KUidCustomInterfaceDevSoundFileBlockLength,
1.75 + aInterface = KMmfPlayCustomInterface for TPlayCustomInterface,
1.76 + aInterface = KMmfRecordCustomInterface for TRecordCustomInterface.
1.77 +@return A pointer to the interface implementation, The return value must be cast to the
1.78 + correct type by the user.
1.79 +*/
1.80 +TAny* CMMFImaAdpcmToPcm16CodecHwDevice::CustomInterface(TUid aInterfaceId)
1.81 + {
1.82 + if(aInterfaceId == KUidCustomInterfaceDevSoundFileBlockLength)
1.83 + {
1.84 + return static_cast<MMMFDevSoundCustomInterfaceFileBlockLength*>(this);
1.85 + }
1.86 + else
1.87 + {
1.88 + return CMMFSwCodecWrapper::CustomInterface(aInterfaceId);
1.89 + }
1.90 + }
1.91 +
1.92 +/**
1.93 +@see CMMFSwCodecWrapper::Start
1.94 +
1.95 +this function sets SampleRate and Channels for CMMFImaAdpcmToPcm16Codec
1.96 +*/
1.97 +TInt CMMFImaAdpcmToPcm16CodecHwDevice::Start(TDeviceFunc aFuncCmd, TDeviceFlow aFlowCmd)
1.98 + {
1.99 + TInt err = static_cast<CMMFImaAdpcmToPcm16Codec*>(iCodec)->Configure(iChannels, iSampleRate, iBlockAlign);
1.100 + if (err == KErrNone)
1.101 + {
1.102 + err = CMMFSwCodecWrapper::Start(aFuncCmd, aFlowCmd);
1.103 + }
1.104 +
1.105 + return err;
1.106 + }
1.107 +
1.108 +/**
1.109 +SetFileBlockLength
1.110 +
1.111 +This function sets file's block length for CMMFImaAdpcmToPcm16CodecHwDevice
1.112 +
1.113 +@param aBlockAlign
1.114 + The file's block length
1.115 +*/
1.116 +void CMMFImaAdpcmToPcm16CodecHwDevice::SetFileBlockLength(TUint aBlockAlign)
1.117 + {
1.118 + iBlockAlign = aBlockAlign;
1.119 + }
1.120 +
1.121 +CMMFImaAdpcmToPcm16Codec::CMMFImaAdpcmToPcm16Codec()
1.122 + {
1.123 + iChannels = 1;
1.124 + iSampleRate = 0;
1.125 + iBlockAlign = KImaAdpcmBlockAlign;
1.126 + iSamplesPerBlock = KImaAdpcmSamplesPerBlock;
1.127 + }
1.128 +/**
1.129 +*
1.130 +* ProcessL
1.131 +* @param aSrc
1.132 +* @param aDst
1.133 +* @pre position of buffer aSrc is 0
1.134 +* @pre position of buffer aDst is 0
1.135 +* @pre sufficient bytes in output to consume input
1.136 +* @return TCodecProcessResult
1.137 +* This function converts IMA ADPCM samples to PCM samples.
1.138 +*
1.139 +*/
1.140 +CMMFSwCodec::TCodecProcessResult CMMFImaAdpcmToPcm16Codec::ProcessL(const CMMFBuffer& aSrc, CMMFBuffer& aDst)
1.141 + {
1.142 + CMMFSwCodec::TCodecProcessResult result;
1.143 + result.iCodecProcessStatus = TCodecProcessResult::EProcessComplete;
1.144 +
1.145 + //convert from generic CMMFBuffer to CMMFDataBuffer
1.146 + const CMMFDataBuffer* src = STATIC_CAST(const CMMFDataBuffer*, &aSrc);
1.147 + CMMFDataBuffer* dst = STATIC_CAST(CMMFDataBuffer*, &aDst);
1.148 +
1.149 + if( !CheckPreconditions( src, dst ) )
1.150 + {
1.151 + //[ precondition(s) violation ]
1.152 + User::Leave(KErrArgument);
1.153 + }
1.154 +
1.155 + //calculate how much source is required to fill the destination buffer
1.156 + TUint blocksRemaining = src->Data().Length() / iBlockAlign;
1.157 +
1.158 + //we need to cast away CONST even on the source, as the TClass needs a TUint8*
1.159 + TUint8* pSrc = CONST_CAST(TUint8*,src->Data().Ptr());
1.160 + TUint8* pDst = CONST_CAST(TUint8*,dst->Data().Ptr());
1.161 +
1.162 + //[ [process full blocks ]
1.163 + TUint dstBytesAdded = 0;
1.164 + for( TUint count = 0; count < blocksRemaining; count++ )
1.165 + {
1.166 + iImaAdpcmTo16Pcm.Convert(pSrc, pDst, iSamplesPerBlock);
1.167 + pSrc += iBlockAlign;
1.168 + pDst += (iSamplesPerBlock * sizeof(TInt16));
1.169 + dstBytesAdded += (iSamplesPerBlock * sizeof(TInt16));
1.170 + }
1.171 +
1.172 + result.iCodecProcessStatus = TCodecProcessResult::EProcessComplete;
1.173 + result.iSrcBytesProcessed = blocksRemaining * iBlockAlign;
1.174 + result.iDstBytesAdded = dstBytesAdded;
1.175 + dst->Data().SetLength(result.iDstBytesAdded);
1.176 +
1.177 + //[ check post conditions
1.178 + __ASSERT_DEBUG( (src->Position() == 0), TMmfAudioCodecPanicsNameSpace::Panic( TMmfAudioCodecPanicsNameSpace::EPostConditionViolation ));
1.179 + __ASSERT_DEBUG( (dst->Position() == 0), TMmfAudioCodecPanicsNameSpace::Panic( TMmfAudioCodecPanicsNameSpace::EPostConditionViolation ));
1.180 + TInt r1 = src->Data().Length();
1.181 + r1 /= iBlockAlign;
1.182 + TInt r2 = dst->Data().Length();
1.183 + r2 /=(iSamplesPerBlock * sizeof(TInt16));
1.184 + __ASSERT_DEBUG( r1== r2, TMmfAudioCodecPanicsNameSpace::Panic(TMmfAudioCodecPanicsNameSpace::EPostConditionViolation ));
1.185 + __ASSERT_DEBUG( dst->Data().Length() % 2 == 0, TMmfAudioCodecPanicsNameSpace::Panic( TMmfAudioCodecPanicsNameSpace::EPostConditionViolation )); // pcm output
1.186 +
1.187 + return result;
1.188 + }
1.189 +
1.190 +/**
1.191 +*
1.192 +* Preconditions
1.193 +* This methos tests the preconditions of the ProcessL method
1.194 +* @return TBool ETrue for sucess and EFalse for failure of the preconditions
1.195 +*
1.196 +**/
1.197 +TBool CMMFImaAdpcmToPcm16Codec::CheckPreconditions( const CMMFDataBuffer* aSrcBuffer, CMMFDataBuffer* aDestBuffer )
1.198 + {
1.199 + TBool result = EFalse;
1.200 +
1.201 + if(! aSrcBuffer )
1.202 + {
1.203 + return result;
1.204 + }
1.205 +
1.206 + if( ! aDestBuffer )
1.207 + {
1.208 + return result;
1.209 + }
1.210 +
1.211 + // Check position of src and dest are 0
1.212 + if( aSrcBuffer->Position() )
1.213 + {
1.214 + return result;
1.215 + }
1.216 +
1.217 + // Check position of src and dest are 0
1.218 + if( aDestBuffer->Position() )
1.219 + {
1.220 + return result;
1.221 + }
1.222 +
1.223 + // check there are sufficient bytes in the output to consume the input
1.224 + const TUint KTempBufferSize = iSamplesPerBlock * 2;
1.225 + TInt numInputSubFrames = aSrcBuffer->Data().Length() / iBlockAlign;
1.226 + TInt numOutputSubFrames = aDestBuffer->Data().MaxLength() / KTempBufferSize;
1.227 +
1.228 + //[ we need modulo 1010 bytes on all src frames that are not the last
1.229 + // frame
1.230 + // For the last frame we will code only whole frames and effectively
1.231 + // drop any remaining samples]
1.232 + TBool validInputDataLength = (aSrcBuffer->Data().Length() % iBlockAlign == 0) ;
1.233 +
1.234 + if( (numInputSubFrames > numOutputSubFrames) || // sufficient space in the output for the input
1.235 + (aSrcBuffer->Position() > 0 ) || // position must be zero since we can eat all the data
1.236 + (aDestBuffer->Position() > 0 ) ||
1.237 + (!validInputDataLength)) //position must be zero
1.238 + {
1.239 + return result;
1.240 + }
1.241 +
1.242 + result = ETrue; // preconditions have been satisfied
1.243 +
1.244 + return result;
1.245 + }
1.246 +
1.247 +/**
1.248 +Configure
1.249 +
1.250 +This function sets file's block length, channels & sample rate.
1.251 +
1.252 +@param aChannels
1.253 + The file's number of channels
1.254 +@param aSampleRate
1.255 + The file's sample rate
1.256 +@param aBlockAlign
1.257 + The file's block length
1.258 +*/
1.259 +TInt CMMFImaAdpcmToPcm16Codec::Configure(TUint aChannels, TUint aSampleRate, TUint aBlockAlign)
1.260 + {
1.261 + iChannels = aChannels;
1.262 + iSampleRate = aSampleRate;
1.263 +
1.264 + if (aBlockAlign < 256)
1.265 + {
1.266 + switch (iSampleRate * iChannels)
1.267 + {
1.268 + case 8000: // fall through, same as 11025
1.269 + case 11025:
1.270 + case 16000:
1.271 + iBlockAlign = 256;
1.272 + break;
1.273 + case 22050:
1.274 + iBlockAlign = 512;
1.275 + break;
1.276 +
1.277 + case 44100:
1.278 + iBlockAlign = 1024;
1.279 + break;
1.280 +
1.281 + case 88200:
1.282 + iBlockAlign = 2048;
1.283 + break;
1.284 +
1.285 + default:
1.286 + return KErrArgument;
1.287 + }
1.288 + }
1.289 + else
1.290 + {
1.291 + iBlockAlign = aBlockAlign;
1.292 + }
1.293 +
1.294 + const TUint KImaAdpcmBitsPerSample = 4;
1.295 + iSamplesPerBlock = (iBlockAlign - 4 * iChannels) * 8 / (KImaAdpcmBitsPerSample * iChannels) + 1;
1.296 +
1.297 + return KErrNone;
1.298 + }
1.299 +
1.300 +
1.301 +
1.302 +
1.303 +
1.304 +