1.1 --- /dev/null Thu Jan 01 00:00:00 1970 +0000
1.2 +++ b/os/mm/devsound/devsoundrefplugin/src/plugin/audio/MMFpcm16ToPcm16HwDevice.cpp Fri Jun 15 03:10:57 2012 +0200
1.3 @@ -0,0 +1,281 @@
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 "MMFpcm16ToPcm16HwDevice.h"
1.20 +
1.21 +
1.22 +/**
1.23 + *
1.24 + * Returns the created hw device for passing audio through audio.
1.25 + * for the wins implementation this would always be pcm16 although
1.26 + * this is effectively a null hw device that will pass any datatype through
1.27 + * @return "CMMFPcm16ToPcm16HwDevice"
1.28 + *
1.29 + */
1.30 +CMMFPcm16ToPcm16HwDevice* CMMFPcm16ToPcm16HwDevice::NewL()
1.31 + {
1.32 + CMMFPcm16ToPcm16HwDevice* self = new (ELeave) CMMFPcm16ToPcm16HwDevice();
1.33 + CleanupStack::PushL(self);
1.34 + self->ConstructL();
1.35 + CleanupStack::Pop(self);
1.36 + return self;
1.37 + }
1.38 +
1.39 +/**
1.40 + *
1.41 + * Second phase constructor.
1.42 + *
1.43 + */
1.44 +void CMMFPcm16ToPcm16HwDevice::ConstructL()
1.45 + {
1.46 + iCodec = new (ELeave) CMMFPcm16ToPcm16Codec();
1.47 + static_cast<CMMFPcm16ToPcm16Codec*>(iCodec)->SetHwDevice(this);
1.48 + }
1.49 +
1.50 +/**
1.51 +*
1.52 +* ~CMMFPcm16ToPcm16HwDevice
1.53 +*
1.54 +**/
1.55 +CMMFPcm16ToPcm16HwDevice::~CMMFPcm16ToPcm16HwDevice()
1.56 + {
1.57 + }
1.58 +
1.59 +/**
1.60 +*
1.61 +* Codec
1.62 +* @return CMMFSwCodec&
1.63 +**/
1.64 +CMMFSwCodec& CMMFPcm16ToPcm16HwDevice::Codec()
1.65 + {
1.66 + return *iCodec;
1.67 + }
1.68 +
1.69 +
1.70 +
1.71 +
1.72 +
1.73 +/**
1.74 +*
1.75 +* ProcessL
1.76 +* @param aSrc Source Buffer
1.77 +* @param sDest Destintion Buffer
1.78 +* @return CMMFSwCodec::TCodecProcessResult
1.79 +*
1.80 +**/
1.81 +CMMFSwCodec::TCodecProcessResult CMMFPcm16ToPcm16Codec::ProcessL(const CMMFBuffer& /*aSource*/, CMMFBuffer& /*aDest*/)
1.82 + {//no processing required for null codec
1.83 + User::Leave(KErrNotSupported);
1.84 + //to keep compiler happy
1.85 + TCodecProcessResult result;
1.86 + result.iCodecProcessStatus = TCodecProcessResult::EEndOfData;
1.87 + result.iSrcBytesProcessed = 0;
1.88 + result.iDstBytesAdded = 0;
1.89 + return result;
1.90 + };
1.91 +
1.92 +
1.93 +TUint CMMFPcm16ToPcm16Codec::SourceBufferSize()
1.94 + {
1.95 + if (!iBufferSize)
1.96 + iBufferSize = iHwDevice->CalculateBufferSize();
1.97 + return iBufferSize;
1.98 + }
1.99 +
1.100 +
1.101 +TUint CMMFPcm16ToPcm16Codec::SinkBufferSize()
1.102 + {
1.103 + if (!iBufferSize)
1.104 + iBufferSize = iHwDevice->CalculateBufferSize();
1.105 + return iBufferSize;
1.106 + }
1.107 +
1.108 +void CMMFPcm16ToPcm16Codec::SetHwDevice(CMMFPcm16ToPcm16HwDevice* aHwDevice)
1.109 + {
1.110 + iHwDevice = aHwDevice;
1.111 + }
1.112 +
1.113 +TUint CMMFPcm16ToPcm16HwDevice::CalculateBufferSize()
1.114 + {
1.115 + TUint sampleRate = 0;
1.116 + TUint channels = 0;
1.117 + TInt useBufferOfSize = 0;
1.118 + TInt minBufferSize = 0;
1.119 + TInt maxBufferSize = 0;
1.120 +
1.121 + if (iPlayCustomInterface)
1.122 + {
1.123 + sampleRate = iSampleRate;
1.124 + channels = iChannels;
1.125 + if ((sampleRate) && (channels))
1.126 + {
1.127 + RMdaDevSound::TSoundFormatsSupportedBuf playFormatsSupported;
1.128 + if (iDataPath->Device().Handle())
1.129 + {
1.130 + iDataPath->Device().PlayFormatsSupported(playFormatsSupported);
1.131 + minBufferSize = playFormatsSupported().iMinBufferSize;
1.132 + maxBufferSize = playFormatsSupported().iMaxBufferSize;
1.133 + }
1.134 + else
1.135 + {//try to get handle
1.136 + TInt err = iDataPath->Device().Open();
1.137 + if (err == KErrNone)
1.138 + {
1.139 + iDataPath->Device().PlayFormatsSupported(playFormatsSupported);
1.140 + minBufferSize = playFormatsSupported().iMinBufferSize;
1.141 + maxBufferSize = playFormatsSupported().iMaxBufferSize;
1.142 + iDataPath->Device().Close();
1.143 + }
1.144 + }
1.145 + }
1.146 + }
1.147 + if ((iRecordCustomInterface) && (!sampleRate) && (!channels))
1.148 + { //must be record
1.149 + sampleRate = iSampleRate;
1.150 + channels = iChannels;
1.151 + if ((sampleRate) && (channels))
1.152 + {//get max and min supported buffer sizes supported by hw
1.153 + MSwSetParamInterface* interface =
1.154 + static_cast<MSwSetParamInterface*>(iDataPath->CustomInterface(KUidSwSetParamInterface));
1.155 + if (interface)
1.156 + {
1.157 + interface->GetBufferSizes(minBufferSize, maxBufferSize);
1.158 + }
1.159 + }
1.160 + }
1.161 +// else convert so not applicable
1.162 +
1.163 + if ((sampleRate) && (channels))
1.164 + {
1.165 + // Buffer size = (SampleRate * BytesPerSample * Channels) / 4
1.166 + useBufferOfSize = ((sampleRate * 2 * channels)/KDevSoundFramesPerSecond + (KDevSoundDeltaFrameSize-1)) &~ (KDevSoundDeltaFrameSize-1);
1.167 + //clamp buffer to desired limits
1.168 + if(useBufferOfSize < KDevSoundMinFrameSize)
1.169 + useBufferOfSize = KDevSoundMinFrameSize;
1.170 + else if(useBufferOfSize > KDevSoundMaxFrameSize)
1.171 + useBufferOfSize = KDevSoundMaxFrameSize;
1.172 +
1.173 + //clamp buffer to limits of hardware
1.174 + if (maxBufferSize)
1.175 + {//buffer size limits have been set by sound driver
1.176 + //check we are within the limits
1.177 + if(useBufferOfSize < minBufferSize)
1.178 + useBufferOfSize = minBufferSize;
1.179 + else if(useBufferOfSize > maxBufferSize)
1.180 + useBufferOfSize = maxBufferSize;
1.181 + }
1.182 + }
1.183 + else
1.184 + {
1.185 + useBufferOfSize = KPCM16ToPCM16BufferSize;
1.186 + }
1.187 +
1.188 + return useBufferOfSize;
1.189 + }
1.190 +
1.191 +TAny* CMMFPcm16ToPcm16HwDevice::CustomInterface(TUid aInterfaceId)
1.192 + {
1.193 + // if this is the bitrate interface then
1.194 + // we support this natively
1.195 + if (aInterfaceId == KUidCustomInterfaceDevSoundBitRate)
1.196 + {
1.197 + return static_cast<MMMFDevSoundCustomInterfaceBitRate*> (this);
1.198 + }
1.199 + else
1.200 + {
1.201 + // otherwise pass the interface call onto the base class
1.202 + return CMMFSwCodecWrapper::CustomInterface(aInterfaceId);
1.203 + }
1.204 + }
1.205 +
1.206 +void CMMFPcm16ToPcm16HwDevice::GetSupportedBitRatesL(RArray<TInt>& aSupportedBitRates)
1.207 + {
1.208 + // precondition of needing the datapath
1.209 + if (!iDataPath)
1.210 + {
1.211 + User::Leave(KErrNotReady);
1.212 + }
1.213 +
1.214 + // we only use this interface on the record interface
1.215 + if (iRecordCustomInterface)
1.216 + {
1.217 + MSwInfoInterface* interface =
1.218 + static_cast<MSwInfoInterface*>(iDataPath->CustomInterface(KUidSwInfoInterface));
1.219 + if (interface)
1.220 + {
1.221 + RArray<TInt> supportedSampleRates;
1.222 + CleanupClosePushL(supportedSampleRates);
1.223 + User::LeaveIfError(interface->GetSupportedSampleRates(supportedSampleRates));
1.224 + BitRatesFromSampleRatesL(aSupportedBitRates, supportedSampleRates);
1.225 + CleanupStack::PopAndDestroy(&supportedSampleRates);
1.226 + }
1.227 + else
1.228 + {
1.229 + User::Leave(KErrNotSupported);
1.230 + }
1.231 + }
1.232 + }
1.233 +
1.234 +void CMMFPcm16ToPcm16HwDevice::BitRatesFromSampleRatesL(RArray<TInt>& aSupportedBitRates,
1.235 + const RArray<TInt>& aSupportedSampleRates)
1.236 + {
1.237 + for (TInt index=0; index<aSupportedSampleRates.Count(); index++)
1.238 + {
1.239 + TInt sampleRate = aSupportedSampleRates[index];
1.240 + TInt bitRate = sampleRate * iChannels * 16;
1.241 + aSupportedBitRates.AppendL(bitRate);
1.242 + }
1.243 + }
1.244 +
1.245 +TInt CMMFPcm16ToPcm16HwDevice::BitRateL()
1.246 + {
1.247 + /*
1.248 + inline void GetRecordFormat(TCurrentSoundFormatBuf& aFormat);
1.249 + inline TInt SetRecordFormat(const TCurrentSoundFormatBuf& aFormat);
1.250 + */
1.251 + TInt bitRate = 0;
1.252 +
1.253 + // need a number of channels and sample rate
1.254 + if (!iSampleRate || !iChannels)
1.255 + {
1.256 + User::Leave(KErrNotReady);
1.257 + }
1.258 +
1.259 + // should be able to just return sample rate * channels * 16
1.260 + if (iRecordCustomInterface)
1.261 + {
1.262 + bitRate = iSampleRate * iChannels * 16;
1.263 + }
1.264 + return bitRate;
1.265 + }
1.266 +
1.267 +void CMMFPcm16ToPcm16HwDevice::SetBitRateL(TInt aBitRate)
1.268 + {
1.269 + // need a datapath and number of channels
1.270 + if (!iDataPath || (!iChannels))
1.271 + {
1.272 + User::Leave(KErrNotReady);
1.273 + }
1.274 +
1.275 + // calculate the correct sample rate but don't set it yet
1.276 + TInt sampleRate = aBitRate / 16 / iChannels;
1.277 +
1.278 + // if the datapath is not playing try and make change immediately
1.279 + if (iDataPath->State() != CMMFSwCodecDataPath::EPlaying)
1.280 + {
1.281 + // update the member variable, assume it is picked up on next record.
1.282 + iSampleRate = sampleRate;
1.283 + }
1.284 + }