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 "MMFpcm16ToPcm16HwDevice.h"
21 * Returns the created hw device for passing audio through audio.
22 * for the wins implementation this would always be pcm16 although
23 * this is effectively a null hw device that will pass any datatype through
24 * @return "CMMFPcm16ToPcm16HwDevice"
27 CMMFPcm16ToPcm16HwDevice* CMMFPcm16ToPcm16HwDevice::NewL()
29 CMMFPcm16ToPcm16HwDevice* self = new (ELeave) CMMFPcm16ToPcm16HwDevice();
30 CleanupStack::PushL(self);
32 CleanupStack::Pop(self);
38 * Second phase constructor.
41 void CMMFPcm16ToPcm16HwDevice::ConstructL()
43 iCodec = new (ELeave) CMMFPcm16ToPcm16Codec();
44 static_cast<CMMFPcm16ToPcm16Codec*>(iCodec)->SetHwDevice(this);
49 * ~CMMFPcm16ToPcm16HwDevice
52 CMMFPcm16ToPcm16HwDevice::~CMMFPcm16ToPcm16HwDevice()
59 * @return CMMFSwCodec&
61 CMMFSwCodec& CMMFPcm16ToPcm16HwDevice::Codec()
73 * @param aSrc Source Buffer
74 * @param sDest Destintion Buffer
75 * @return CMMFSwCodec::TCodecProcessResult
78 CMMFSwCodec::TCodecProcessResult CMMFPcm16ToPcm16Codec::ProcessL(const CMMFBuffer& /*aSource*/, CMMFBuffer& /*aDest*/)
79 {//no processing required for null codec
80 User::Leave(KErrNotSupported);
81 //to keep compiler happy
82 TCodecProcessResult result;
83 result.iCodecProcessStatus = TCodecProcessResult::EEndOfData;
84 result.iSrcBytesProcessed = 0;
85 result.iDstBytesAdded = 0;
90 TUint CMMFPcm16ToPcm16Codec::SourceBufferSize()
93 iBufferSize = iHwDevice->CalculateBufferSize();
98 TUint CMMFPcm16ToPcm16Codec::SinkBufferSize()
101 iBufferSize = iHwDevice->CalculateBufferSize();
105 void CMMFPcm16ToPcm16Codec::SetHwDevice(CMMFPcm16ToPcm16HwDevice* aHwDevice)
107 iHwDevice = aHwDevice;
110 TUint CMMFPcm16ToPcm16HwDevice::CalculateBufferSize()
112 TUint sampleRate = 0;
114 TInt useBufferOfSize = 0;
115 TInt minBufferSize = 0;
116 TInt maxBufferSize = 0;
118 if (iPlayCustomInterface)
120 sampleRate = iSampleRate;
121 channels = iChannels;
122 if ((sampleRate) && (channels))
124 RMdaDevSound::TSoundFormatsSupportedBuf playFormatsSupported;
125 if (iDataPath->Device().Handle())
127 iDataPath->Device().PlayFormatsSupported(playFormatsSupported);
128 minBufferSize = playFormatsSupported().iMinBufferSize;
129 maxBufferSize = playFormatsSupported().iMaxBufferSize;
133 TInt err = iDataPath->Device().Open();
136 iDataPath->Device().PlayFormatsSupported(playFormatsSupported);
137 minBufferSize = playFormatsSupported().iMinBufferSize;
138 maxBufferSize = playFormatsSupported().iMaxBufferSize;
139 iDataPath->Device().Close();
144 if ((iRecordCustomInterface) && (!sampleRate) && (!channels))
146 sampleRate = iSampleRate;
147 channels = iChannels;
148 if ((sampleRate) && (channels))
149 {//get max and min supported buffer sizes supported by hw
150 MSwSetParamInterface* interface =
151 static_cast<MSwSetParamInterface*>(iDataPath->CustomInterface(KUidSwSetParamInterface));
154 interface->GetBufferSizes(minBufferSize, maxBufferSize);
158 // else convert so not applicable
160 if ((sampleRate) && (channels))
162 // Buffer size = (SampleRate * BytesPerSample * Channels) / 4
163 useBufferOfSize = ((sampleRate * 2 * channels)/KDevSoundFramesPerSecond + (KDevSoundDeltaFrameSize-1)) &~ (KDevSoundDeltaFrameSize-1);
164 //clamp buffer to desired limits
165 if(useBufferOfSize < KDevSoundMinFrameSize)
166 useBufferOfSize = KDevSoundMinFrameSize;
167 else if(useBufferOfSize > KDevSoundMaxFrameSize)
168 useBufferOfSize = KDevSoundMaxFrameSize;
170 //clamp buffer to limits of hardware
172 {//buffer size limits have been set by sound driver
173 //check we are within the limits
174 if(useBufferOfSize < minBufferSize)
175 useBufferOfSize = minBufferSize;
176 else if(useBufferOfSize > maxBufferSize)
177 useBufferOfSize = maxBufferSize;
182 useBufferOfSize = KPCM16ToPCM16BufferSize;
185 return useBufferOfSize;
188 TAny* CMMFPcm16ToPcm16HwDevice::CustomInterface(TUid aInterfaceId)
190 // if this is the bitrate interface then
191 // we support this natively
192 if (aInterfaceId == KUidCustomInterfaceDevSoundBitRate)
194 return static_cast<MMMFDevSoundCustomInterfaceBitRate*> (this);
198 // otherwise pass the interface call onto the base class
199 return CMMFSwCodecWrapper::CustomInterface(aInterfaceId);
203 void CMMFPcm16ToPcm16HwDevice::GetSupportedBitRatesL(RArray<TInt>& aSupportedBitRates)
205 // precondition of needing the datapath
208 User::Leave(KErrNotReady);
211 // we only use this interface on the record interface
212 if (iRecordCustomInterface)
214 MSwInfoInterface* interface =
215 static_cast<MSwInfoInterface*>(iDataPath->CustomInterface(KUidSwInfoInterface));
218 RArray<TInt> supportedSampleRates;
219 CleanupClosePushL(supportedSampleRates);
220 User::LeaveIfError(interface->GetSupportedSampleRates(supportedSampleRates));
221 BitRatesFromSampleRatesL(aSupportedBitRates, supportedSampleRates);
222 CleanupStack::PopAndDestroy(&supportedSampleRates);
226 User::Leave(KErrNotSupported);
231 void CMMFPcm16ToPcm16HwDevice::BitRatesFromSampleRatesL(RArray<TInt>& aSupportedBitRates,
232 const RArray<TInt>& aSupportedSampleRates)
234 for (TInt index=0; index<aSupportedSampleRates.Count(); index++)
236 TInt sampleRate = aSupportedSampleRates[index];
237 TInt bitRate = sampleRate * iChannels * 16;
238 aSupportedBitRates.AppendL(bitRate);
242 TInt CMMFPcm16ToPcm16HwDevice::BitRateL()
245 inline void GetRecordFormat(TCurrentSoundFormatBuf& aFormat);
246 inline TInt SetRecordFormat(const TCurrentSoundFormatBuf& aFormat);
250 // need a number of channels and sample rate
251 if (!iSampleRate || !iChannels)
253 User::Leave(KErrNotReady);
256 // should be able to just return sample rate * channels * 16
257 if (iRecordCustomInterface)
259 bitRate = iSampleRate * iChannels * 16;
264 void CMMFPcm16ToPcm16HwDevice::SetBitRateL(TInt aBitRate)
266 // need a datapath and number of channels
267 if (!iDataPath || (!iChannels))
269 User::Leave(KErrNotReady);
272 // calculate the correct sample rate but don't set it yet
273 TInt sampleRate = aBitRate / 16 / iChannels;
275 // if the datapath is not playing try and make change immediately
276 if (iDataPath->State() != CMMFSwCodecDataPath::EPlaying)
278 // update the member variable, assume it is picked up on next record.
279 iSampleRate = sampleRate;