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 <mmf/server/mmfswcodecwrapper.h>
17 #include "mmfSwCodecPlayDataPath.h"
18 #include "mmfSwCodecRecordDataPath.h"
19 #include "mmfSwCodecConvertDataPath.h"
20 #include <mmf/server/mmfswcodecwrappercustominterfacesuids.hrh>
21 #include "mmfswcodecwrapperCustomInterfaces.h"
22 #include <mmf/common/mmfpaniccodes.h>
30 void Panic(TInt aPanicCode)
32 _LIT(KMMFSwCodecWrapperPanicCategory, "MMFSwCodecWrapper");
33 User::Panic(KMMFSwCodecWrapperPanicCategory, aPanicCode);
38 * This method is not be exported as it is only
39 * intended to be called within this DLL.
40 * It's purpose is to assign an RMdaDevSound to the play
44 void TPlayCustomInterface::SetDevice(RMdaDevSound* aDevice)
49 void TPlayCustomInterface::SetVolume(TUint aVolume)
52 if (iDevice && iDevice->Handle())
53 iDevice->SetPlayVolume(iVolume);
57 * Procedure to get the number of bytes played by the device driver
58 * If there is no handle available to the device driver then the
59 * procedure returns the last known value
61 * @return number of bytes played
63 TUint TPlayCustomInterface::BytesPlayed()
67 if (iDevice->Handle())
68 iBytesPlayed = iDevice->BytesPlayed();
74 * Procedure to get the number of bytes recorded by the device
76 * @return The number of bytes recorded by an existing datapath. If there
77 * is no datapath, then the last known number of bytes recorded will be returned.
79 TUint TRecordCustomInterface::BytesRecorded()
83 iBytesRecorded = iDataPath->RecordedBytesCount();
85 return iBytesRecorded;
91 EXPORT_C CMMFSwCodecWrapper::CMMFSwCodecWrapper()
98 The destructor is called by ECom framework allowing derived classes
99 to clean up implementation specific resources. The sound
100 device drivers are freed.
102 EXPORT_C CMMFSwCodecWrapper::~CMMFSwCodecWrapper()
106 delete iPlayCustomInterface;
107 delete iRecordCustomInterface;
111 Initializes the hardware device tasks - in the case of a
112 sw codec wrapper 'hardware device' this consits of loading the
113 sound device drivers and creating the CMMFSwCodec.
116 Device initialization parameters.
117 Only the iHwDeviceObserver is used for CMFSwCodecWrapper
118 derived CMMFHwDevices.
119 @return An error code indicating if the function call was successful. KErrNone on success, otherwise
120 another of the system-wide error codes.
122 EXPORT_C TInt CMMFSwCodecWrapper::Init(THwDeviceInitParams &aDevInfo)
125 // [ precondition that aDevInfo has a valid observer ]
126 if (!aDevInfo.iHwDeviceObserver)
129 iHwDeviceObserver = aDevInfo.iHwDeviceObserver;
130 #ifndef SYMBIAN_MDF_SHAREDCHUNK_SOUNDDRIVER //Adapter loads the drivers
131 // Try to load the audio physical driver
132 TInt ret = User::LoadPhysicalDevice(KPddFileName);
133 if ((ret!=KErrNone) && (ret!=KErrAlreadyExists))
136 // Try to load the audio logical driver
137 ret = User::LoadLogicalDevice(KLddFileName);
138 if ((ret!=KErrNone) && (ret!=KErrAlreadyExists))
141 iCodec = &(Codec()); //create codec
143 //[ assert the post condition ]
145 return KErrNotSupported;
152 Starts Encoding or Decoding task(s) based on the parameter specified.
155 The device function specifying the requested service i.e. decode or encode
156 where EDevEncode = Record, EDevDecode = Play and EDevNullFunc = Convert.
158 The device flow directions for requested service.
159 This parameter is ignored for CMMFSwCodecWrapper CMMFHwDevicePlugins
160 @return An error code indicating if the function call was successful. KErrNone on success, otherwise
161 another of the system-wide error codes.
163 EXPORT_C TInt CMMFSwCodecWrapper::Start(TDeviceFunc aFuncCmd, TDeviceFlow /*aFlowCmd*/)
165 TInt error = KErrNone;
167 // [ precondition that aFuncCmd is valid]
168 if (!((aFuncCmd == EDevEncode)|(aFuncCmd == EDevDecode)|(aFuncCmd == EDevNullFunc)))
171 // [ precondition that iCodec is present]
173 return KErrNotReady; //make sure the codec has been added
177 case EDevEncode: // Audio record
179 error = StartEncode();
182 case EDevDecode: // Audio play
184 error = StartDecode();
187 case EDevNullFunc: //Audio Convert
189 error = StartConvert();
193 error = KErrNotSupported;
197 //[ assert the post conditions ]
200 {//only assert if no error otherwise post consitions not valid
201 __ASSERT_DEBUG(iDataPath, Panic(EMMFSwCodecWrapperNoDataPath));
202 if ((aFuncCmd == EDevEncode)||(aFuncCmd == EDevDecode))
203 __ASSERT_DEBUG(iDataPath->Device().Handle(), Panic(EMMFSwCodecWrapperNoDevice));
206 if(error != KErrNone && iDataPath && aFuncCmd!=EDevEncode)
207 {//if error happens after opening LDD close it
208 if (iDataPath->Device().Handle()!= KNullHandle)
210 iDataPath->Device().Close();
218 TInt CMMFSwCodecWrapper::StartDecode()
220 TInt error = KErrNone;
222 //[ assert precondition that play custom interface is present]
223 //if there is no play custom interface then the user of the CMMFSwCodecWrapper
224 //cannot have set any of the custom settings such as sample rate.
225 if (!iPlayCustomInterface)
231 TRAP(error,iDataPath = CMMFSwCodecPlayDataPath::NewL());
232 //if datapath could not be created, return error code
233 if (error != KErrNone)
238 //here we are sure iDataPath has been correctly allocated
239 iDataPath->SetObserver(*iHwDeviceObserver);
240 error = iDataPath->AddCodec(*iCodec);
241 if (error == KErrNone)
243 iDeviceBufferSize = (iCodec->SinkBufferSize());
244 static_cast<CMMFSwCodecPlayDataPath*>(iDataPath)->SetPlayCustomInterface(*iPlayCustomInterface);
248 // if could not add codec to datapath, return error code
253 //Here we know that error is KerrNone, now we can check the state of the datapath
254 if (iDataPath->State() != CMMFSwCodecDataPath::EPlaying)
255 {//datapath was created ok and we are not playing
256 if (iDataPath->State() == CMMFSwCodecDataPath::EStopped)
257 {//starting from 'fresh so set sound device settings
258 if (!iDataPath->Device().Handle())
259 {//if Device() is called then we need a valid sound device handle
260 error = iDataPath->Device().Open();
261 if (error != KErrNone)
264 static_cast<TPlayCustomInterface*>(iPlayCustomInterface)->SetDevice(&(iDataPath->Device()));
265 iDataPath->Device().SetPlayVolume(iPlayCustomInterface->Volume());
266 RMdaDevSound::TCurrentSoundFormatBuf soundDeviceSettings;
267 soundDeviceSettings().iRate = iSampleRate;
268 //this would normally be pcm16
269 soundDeviceSettings().iEncoding = RMdaDevSound::EMdaSoundEncoding16BitPCM;
270 //1 = mono 2 = stereo
271 soundDeviceSettings().iChannels = iChannels;
272 //tell sound driver what buffer size to expect
273 //it is up the the implementor to make use the device can support
274 //the required buffer size
275 soundDeviceSettings().iBufferSize = iDeviceBufferSize;
276 error = iDataPath->Device().SetPlayFormat(soundDeviceSettings);
277 }//iDataPath->State() == CMMFSwCodecDataPath::EStopped
278 //else resuming from pause
279 if ((error == KErrNone)||(error == KErrInUse))
280 error = iDataPath->Start();
281 }//status == KErrNone
286 TInt CMMFSwCodecWrapper::StartEncode()
289 //[ assert precondition that record custom interface is present]
290 //if there is no record custom interface then the user of the CMMFSwCodecWrapper
291 //cannot have set any of the custom settings such as sample rate.
292 if (!iRecordCustomInterface)
295 TInt error = KErrNone;
298 TRAP(error,iDataPath = CMMFSwCodecRecordDataPath::NewL());
299 //if datapath could not be created, return error code
300 if (error != KErrNone)
305 //here we are sure iDataPath has been correctly allocated
306 iDataPath->SetObserver(*iHwDeviceObserver);
307 error = iDataPath->AddCodec(*iCodec);
308 if (error == KErrNone)
310 iDeviceBufferSize = (iCodec->SourceBufferSize());
311 static_cast<TRecordCustomInterface*>(iRecordCustomInterface)->SetDataPath(static_cast<CMMFSwCodecRecordDataPath*>(iDataPath));
315 // if could not add codec to datapath, return error code
320 //Here we know that error is KerrNone, now we can check the state of the datapath
321 if (iDataPath->State() != CMMFSwCodecDataPath::EPlaying)
323 if (iDataPath->State() == CMMFSwCodecDataPath::EStopped)
325 MSwSetParamInterface* setParams =
326 static_cast<MSwSetParamInterface*>(iDataPath->CustomInterface(KUidSwSetParamInterface));
327 ASSERT(!error); // should not get here if error set
328 error = setParams->SetGain(iRecordCustomInterface->Gain());
331 error = setParams->SetNumChannels(iChannels);
335 error = setParams->SetSampleRate(iSampleRate);
338 if (error == KErrNone)
340 error = iDataPath->Start();
347 TInt CMMFSwCodecWrapper::StartConvert()
350 TInt error = KErrNone;
353 TRAP(error,iDataPath = CMMFSwCodecConvertDataPath::NewL());
354 if (error != KErrNone)
360 //Here we know we are not dereferencing a null pointer as iDataPath has been correctly initialised
361 iDataPath->SetObserver(*iHwDeviceObserver);
362 error = iDataPath->AddCodec(*iCodec);
364 if (error == KErrNone)
366 error = iDataPath->Start();
373 Temporarily suspends the current task of decoding or encoding.
375 @return An error code indicating if the function call was successful. KErrNone on success, otherwise
376 another of the system-wide error codes.
378 EXPORT_C TInt CMMFSwCodecWrapper::Pause()
380 // [ precondition that datapath exists ]
389 Stops the current on-going task.
391 @return An error code indicating if the function call was successful. KErrNone on success, otherwise
392 another of the system-wide error codes.
394 EXPORT_C TInt CMMFSwCodecWrapper::Stop()
396 // [ precondition that datapath exists ]
406 Stops and deletes the codec.
408 This default implementation simply calls DeleteCodec() and then Stop()
409 but real hardware devices might use this method to free up resources.
411 @return An error code indicating if the function call was successful. KErrNone on success, otherwise
412 another of the system-wide error codes.
414 EXPORT_C TInt CMMFSwCodecWrapper::StopAndDeleteCodec()
416 TInt stopError = Stop();
417 TInt deleteError = DeleteCodec();
419 if (stopError != KErrNone)
427 This default implementation does nothing
428 but real hardware devices might use this method to free up resources.
429 @return Error code. KErrNone if successful
431 EXPORT_C TInt CMMFSwCodecWrapper::DeleteCodec()
437 Call this function to notify hardware device implementation that
438 data is available in aFillBufferPtr for decoding.
440 @param aFillBufferPtr
441 The data buffer filled by the observer.
443 @return An error code indicating if the function call was successful. KErrNone on success, otherwise
444 another of the system-wide error codes.
446 EXPORT_C TInt CMMFSwCodecWrapper::ThisHwBufferFilled(CMMFBuffer& aFillBufferPtr)
448 TRAPD(err,iDataPath->BufferFilledL(STATIC_CAST(CMMFDataBuffer&, aFillBufferPtr)));
453 Call this function to notify hardware device implementation that
454 data in aEmptyBufferPtr from encoding is processed.
457 The data buffer processed by observer.
459 @return An error code indicating if the function call was successful. KErrNone on success, otherwise
460 another of the system-wide error codes.
462 EXPORT_C TInt CMMFSwCodecWrapper::ThisHwBufferEmptied(CMMFBuffer& aBuffer)
464 TRAPD(err,iDataPath->BufferEmptiedL(STATIC_CAST(CMMFDataBuffer&, aBuffer)));
470 Retrieves a custom interface to the device.
471 The reference CMMFSwCodecWrapper supports two standard custom interfaces,
472 MPlayCustomInterface and MRecordCustomInterface.
475 Interface UID, defined with the custom interface.
476 aInterface = KMmfPlaySettingsCustomInterface for MPlayCustomInterface,
477 aInterface = KMmfRecordSettingsCustomInterface for MRecordCustomInterface.
478 aInterface = KMmfUidEmptyBuffersCustomInterface for MEmptyBuffersCustomInterface
479 Actual device implementations of CMMFSwCodecWrapper may do this differently however.
480 @return A pointer to the interface implementation, or NULL if the device can not
481 implement the interface requested. The return value must be cast to the
482 correct type by the user.
484 EXPORT_C TAny* CMMFSwCodecWrapper::CustomInterface(TUid aInterface)
488 if (aInterface.iUid == KMmfPlaySettingsCustomInterface)
490 if (!iPlayCustomInterface)
491 TRAP(err,iPlayCustomInterface = new(ELeave)TPlayCustomInterface());
495 ret = static_cast<TAny*>(iPlayCustomInterface);
497 else if (aInterface.iUid == KMmfRecordSettingsCustomInterface)
499 if (!iRecordCustomInterface)
500 TRAP(err,iRecordCustomInterface = new(ELeave)TRecordCustomInterface());
504 ret = static_cast<TAny*>(iRecordCustomInterface);
507 else if (aInterface.iUid == KMmfUidEmptyBuffersCustomInterface || aInterface == KTimePlayedCustomInterfaceTypeUid || aInterface == KIgnoreUnderflowCustomInterfaceTypeUid)
515 ret = static_cast<CMMFSwCodecDataPath*>(iDataPath)->CustomInterface(aInterface);
524 Used to configure the sample rate and stereo mode of a CMMFHwDevice plugin.
526 The configuration of HwDevices is device specific and is not used in any of the reference
527 devices that return KErrNotSupported.
530 The device configuration.
532 EXPORT_C TInt CMMFSwCodecWrapper::SetConfig(TTaskConfig& aConfig)
534 if (aConfig.iUid != KUidRefDevSoundTaskConfig)
536 iSampleRate = aConfig.iRate;
538 if (aConfig.iStereoMode == ETaskMono)
542 else if (aConfig.iStereoMode == ETaskInterleaved || aConfig.iStereoMode == ETaskNonInterleaved)
554 Used to set iVbrFlag on the datapath.
556 This method is used to set the iVbrFlag in datapath. This flag is added to datapath to avail the
557 alternative dataflow wherein datapath makes sure that destinationbuffer is filled to its maximum length
558 before sending it to the sound driver. Sending the buffer directly to the device causes underflow incase of VBR codecs.
560 EXPORT_C void CMMFSwCodecWrapper::SetVbrFlag()
564 TUid cUid = TUid::Uid(KSetVbrFlagCustomInterfaceTypeUid);
565 iDataPath->CustomInterface(cUid);