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.
14 // source\server\mmfswcodecplaydatapath.cpp
18 #include "mmfSwCodecPlayDataPath.h"
19 #include <mmf/server/mmfswcodecwrapper.h>
20 #include <mmf/server/mmfswcodecwrappercustominterfacesuids.hrh>
21 #include <mmf/common/mmfpaniccodes.h>
22 #include "mmfSwCodecUtility.h"
24 const TInt KBytesPerSample = 2;
25 const TInt KMaxBytesInSec = 192000; //considering maximum samplerate 96KHz
26 CMMFSwCodecPlayDataPath* CMMFSwCodecPlayDataPath::NewL()
28 CMMFSwCodecPlayDataPath* self = new(ELeave) CMMFSwCodecPlayDataPath;
29 CleanupStack::PushL(self);
36 void CMMFSwCodecPlayDataPath::ConstructL()
38 iAudioPlayer = new (ELeave) CDataPathPlayer(*this,CActive::EPriorityUserInput);
39 iSoundDeviceErrorReceiver = new (ELeave) CSoundDevPlayErrorReceiver(*this, CActive::EPriorityUserInput);
40 iUtility = CMMFSwCodecUtility::NewL();
45 CMMFSwCodecPlayDataPath::~CMMFSwCodecPlayDataPath()
48 delete iSoundDeviceErrorReceiver;
56 if (!iCodec->IsNullCodec())
58 delete iSoundDeviceBuffer;
62 #ifdef __USE_MMF_TRANSFERBUFFERS__
63 delete iTransferWindow;
67 iTransferBuffer->Close();
68 delete iTransferBuffer;
72 #ifdef __USE_MMF_PTRBUFFERS__
73 delete iPtrBufferMemoryBlock;
78 TInt CMMFSwCodecPlayDataPath::SetObserver(MMMFHwDeviceObserver& aObserver)
81 if (iHwDeviceObserver)
83 error = KErrAlreadyExists;
87 iHwDeviceObserver = &aObserver;
94 TInt CMMFSwCodecPlayDataPath::AddCodec(CMMFSwCodec& aCodec)
97 return KErrNotSupported; //doesn't support multiple codecs
103 // Allocate data buffer
104 iSourceBufferSize = iCodec->SourceBufferSize();
105 iSoundDevBufferSize = iCodec->SinkBufferSize();
107 if ((!iSourceBufferSize)||(!iSoundDevBufferSize))
108 err = KErrArgument; //codec plugin has not specified buffer size
112 #ifdef __USE_MMF_TRANSFERBUFFERS__
113 TRAP(err,iSourceBuffer = CreateTransferBufferL(iSourceBufferSize, static_cast<CMMFTransferBuffer*>(iSourceBuffer)));
115 #ifdef __USE_MMF_PTRBUFFERS__
116 TRAP(err,iSourceBuffer = CreatePtrBufferL(iSourceBufferSize));
118 TRAP(err,iSourceBuffer = CMMFDataBuffer::NewL(iSourceBufferSize));
124 if (iCodec->IsNullCodec())
125 {//use source buffer for sound device buffer
126 iSoundDeviceBuffer = NULL;
129 {//codec needs separate source and sound device buffers
130 TRAP(err,iSoundDeviceBuffer = CMMFDataBuffer::NewL(iSoundDevBufferSize));
137 TInt CMMFSwCodecPlayDataPath::Start()
139 TInt startError = KErrNone;
142 {//check that a codec has been added
143 startError = KErrNotReady;
145 if ((!iSoundDevice.Handle())&&(!startError))
146 {//check that the sound drivers can be opened
147 startError = iSoundDevice.Open();
150 if (iState == EPaused)
151 {//we are paused so need to resume play
155 RDebug::Print(_L("CMMFSwCodecPlayDataPath::Start-Resume"));
157 iAudioPlayer->ResumePlaying();
161 else if (!startError)
164 RDebug::Print(_L("CMMFSwCodecPlayDataPath::Start-Normal"));
166 // get sample rate and channels from RMdaDevSound
167 RMdaDevSound::TCurrentSoundFormatBuf format;
168 iSoundDevice.GetPlayFormat(format);
169 iSampleRate = format().iRate;
170 iChannels = format().iChannels;
172 iNoMoreSourceData = EFalse;
173 iNoMoreSoundDeviceData = EFalse;
174 iSourceBuffer->SetLastBuffer(EFalse);
177 iSoundDeviceErrorReceiver->Start();
178 TRAP(startError,FillSourceBufferL()); //get initial buffer of audio data
179 if (startError == KErrNone)
181 // Start the player objects
182 iAudioPlayer->Start();
185 {//failed to start up correctly go back to stopped state
187 iSoundDeviceErrorReceiver->Stop();
194 // *** Main Play Loop ***
196 void CMMFSwCodecPlayDataPath::FillSourceBufferL()
197 {//asks observer to fill the source buffer
198 // Ask immediately for data from the observer
199 #ifdef __CYCLE_MMF_DATABUFFERS__
200 // Create a new buffer to replicate INC021405 Play-EOF-Play on HwAccelerated solution Panics
201 // If the creation fails, we carry on regardless as the original buffer will not have been
202 // destroyed. Must do this as alloc fail tests will not run.
205 iSourceBuffer = CycleAudioBuffer(iSourceBuffer);
207 #endif // __CYCLE_MMF_DATABUFFERS__
208 User::LeaveIfError(iHwDeviceObserver->FillThisHwBuffer(*iSourceBuffer));
213 void CMMFSwCodecPlayDataPath::BufferFilledL(CMMFDataBuffer& aBuffer)
214 {//call back from observer to indicate buffer has been filled
215 if (iState == EStopped)
216 User::Leave(KErrNotReady);//ok if paused?
218 iSourceBuffer = &aBuffer;
219 iSourceBuffer->SetStatus(EFull);
221 RDebug::Print(_L("CMMFSwCodecPlayDataPath::BufferFilledL"));
224 //need to check that the buffer size is not 0 - if so assume we've reached the end of the data
225 if (!iSourceBuffer->BufferSize())
226 {//no buffer - could be end of source or could be that the source has no data??
227 iNoMoreSourceData = ETrue;
229 RDebug::Print(_L("CMMFSwCodecPlayDataPath::BufferFilledL-NoMoreSourceData"));
232 //even if the buffer size is 0 we still
233 //need to perform the following to get the sound device callback
234 FillSoundDeviceBufferL(); //get buffer in pcm16 format for sound device
236 /* iVbrFlag is added to datapath to avail the alternative dataflow wherein datapath makes sure that
237 destinationbuffer is filled to its maximum length before sending it to the sound driver.
238 Sending the buffer directly to the device causes underflow incase of Vorbis codecs.*/
241 /*There are two cases we need to deal here
242 1. When the output of the codec is 0 for header data.
243 in that case, repeat till actual decoding of ogg packets and pages.
244 2. When destination buffer is not filled even to its half length, get next source buffer
245 and decode it. This is to avoid underflows when ever we receive little pcm for a
246 a given source buffer.
248 if (iSoundDeviceBuffer->Data().Length() < iSoundDeviceBuffer->Data().MaxLength()/2 && !(iSoundDeviceBuffer->LastBuffer()))
250 iSourceBuffer->SetStatus(EAvailable); //source buffer is now available
251 iSoundDeviceBuffer->SetPosition(iSoundDeviceBuffer->Data().Length());//this indicates the available space in the buffer to the codec
255 else //data is sufficient to avoid underflows
257 iSoundDeviceBuffer->SetPosition(0);
258 if(iSoundDeviceBuffer->Data().Length()==0 && iSoundDeviceBuffer->LastBuffer())
260 iNoMoreSoundDeviceData = ETrue;
265 // attenuate the amplitude of the samples if volume ramping has been changed
267 if (iCustomInterface)
269 TTimeIntervalMicroSeconds volumeRamp = iCustomInterface->VolumeRamp();
270 if (volumeRamp != iVolumeRamp)
272 iVolumeRamp = volumeRamp;
273 if (iVolumeRamp.Int64() != 0)
275 iUtility->ConfigAudioRamper(
279 iRampAudioSample = ETrue;
283 iRampAudioSample = EFalse;
287 if (iRampAudioSample)
288 iRampAudioSample = iUtility->RampAudio(iSoundDeviceBuffer);
291 iAudioPlayer->PlayData(*iSoundDeviceBuffer); //play data to sound drivers
293 if (iSourceBuffer->LastBuffer())//check last buffer flag
295 iNoMoreSourceData = ETrue;
297 RDebug::Print(_L("CMMFSwCodecPlayDataPath::BufferFilledL-LBNoMoreSourceData"));
303 void CMMFSwCodecPlayDataPath::FillSoundDeviceBufferL()
304 {//use CMMFSwCodec to fill the sound device buffer
306 CMMFSwCodec::TCodecProcessResult codecProcessResult;
308 if (iCodec->IsNullCodec())
309 {//no codec so data can be sent direct to sink
310 iSoundDeviceBuffer = iSourceBuffer;
311 iSoundDeviceBuffer->SetStatus(EFull); //sink buffer is full
315 //pass buffer to codec for processing
316 codecProcessResult = iCodec->ProcessL(*iSourceBuffer, *iSoundDeviceBuffer);
318 if (iSourceBuffer->LastBuffer()) //if source is last buffer so is sound dev
319 iSoundDeviceBuffer->SetLastBuffer(ETrue);
320 if ((!iSoundDeviceBuffer->BufferSize())&&(codecProcessResult.iDstBytesAdded))
321 {//the codec has added data but not set the buffer length
322 iSoundDeviceBuffer->Data().SetLength(codecProcessResult.iDstBytesAdded);
324 //only supports EProcessComplete
325 switch (codecProcessResult.iCodecProcessStatus)
327 case CMMFSwCodec::TCodecProcessResult::EProcessComplete:
328 //finished procesing source data - all data in sink buffer
330 iSoundDeviceBuffer->SetStatus(EFull); //sink buffer is full
333 #ifdef SYMBIAN_VARIABLE_BITRATE_CODEC
334 case CMMFSwCodec::TCodecProcessResult::EProcessIncomplete:
335 //finished procesing source data - all data in sink buffer
337 iSoundDeviceBuffer->SetStatus(EFull); //sink buffer is full
341 case CMMFSwCodec::TCodecProcessResult::EDstNotFilled:
342 //could be the last buffer in which case dst might not get filled
344 iSoundDeviceBuffer->SetStatus(EFull); //sink buffer is full
347 case CMMFSwCodec::TCodecProcessResult::EEndOfData:
348 //no more data - send what we've got to the sink
349 //note we can't always rely on this - in many cases the codec will not know when
350 //it has reached the end of data.
352 iSoundDeviceBuffer->SetStatus(EFull);//sink buffer may not really be 'full' but its as full as it going to get
353 iNoMoreSourceData = ETrue;
354 //doesn't matter if sink buffer is not full
358 Panic(EMMFSwCodecWrapperBadCodec); //should never get here - bad codec
364 void CMMFSwCodecPlayDataPath::BufferEmptiedL(const CMMFDataBuffer& aBuffer)
365 {//call back from CDataPathPlayer when the sound device buffer has been emptied
366 if (&aBuffer != iSoundDeviceBuffer)
367 Panic(EMMFSwCodecWrapperBadBuffer);
368 if(iVbrFlag && (iSourceBuffer->Status() == EUnAvailable || iNoMoreSourceData))
369 {//No more source data. Play rest of the decoded data.Inform codec not to consider the source buffer
370 if(iSourceBuffer->Status()!=EUnAvailable)
372 iSourceBuffer->SetStatus(EUnAvailable);
374 FillSoundDeviceBufferL();
375 if(iSoundDeviceBuffer->BufferSize() > 0)
377 // attenuate the amplitude of the samples if volume ramping has been changed
379 if (iCustomInterface)
381 TTimeIntervalMicroSeconds volumeRamp = iCustomInterface->VolumeRamp();
382 if (volumeRamp != iVolumeRamp)
384 iVolumeRamp = volumeRamp;
385 if (iVolumeRamp.Int64() != 0)
387 iUtility->ConfigAudioRamper(iVolumeRamp.Int64(), iSampleRate, iChannels);
388 iRampAudioSample = ETrue;
392 iRampAudioSample = EFalse;
396 if (iRampAudioSample)
398 iRampAudioSample = iUtility->RampAudio(iSoundDeviceBuffer);
402 iAudioPlayer->PlayData(*iSoundDeviceBuffer); //play data to sound drivers
407 if(iNoMoreSourceData)
409 iNoMoreSoundDeviceData = ETrue;
411 iSourceBuffer->SetStatus(EAvailable);
414 if (!iNoMoreSourceData)
418 //*** End of Main Play Loop ***
421 void CMMFSwCodecPlayDataPath::Stop()
423 iAudioPlayer->Cancel();
424 iSoundDeviceErrorReceiver->Cancel();
425 iSoundDevice.Close();
427 #ifdef __CYCLE_MMF_DATABUFFERS__
428 // Create a new buffer to replicate INC021405 Play-EOF-Play on HwAccelerated solution Panics
429 // If the creation fails, we carry on regardless as the original buffer will not have been
430 // destroyed. Must do this as alloc fail tests will not run.
433 iSourceBuffer = CycleAudioBuffer(iSourceBuffer);
435 #endif // __CYCLE_MMF_DATABUFFERS__
441 void CMMFSwCodecPlayDataPath::Pause()
443 //since a pause can happen anyway in the datatransfer -need to set to a known
444 //state so that when play is resumed the behaviour is predictable
445 if (iSoundDevice.Handle())
447 iSoundDevice.PausePlayBuffer(); //needs new LDD
450 RDebug::Print(_L("Pause"));
454 {//an error must have occured
460 TInt CMMFSwCodecPlayDataPath::EmptyBuffers()
462 TInt error = KErrNone;
463 if (iSoundDevice.Handle() == 0)
465 error = KErrNotReady;
469 iAudioPlayer->Cancel();
470 iSoundDevice.CancelPlayData();
471 iSoundDeviceErrorReceiver->Stop();
478 RMdaDevSound& CMMFSwCodecPlayDataPath::Device()
484 void CMMFSwCodecPlayDataPath::SoundDeviceException(TInt aError)
488 if(!iVbrFlag && aError==KErrUnderflow && !iNoMoreSourceData)
493 //for VBR codec data,no more source does not mean that no more sounddevice data
494 //so ignore underflows till the last buffer is played from the codec
495 else if(iVbrFlag && aError==KErrUnderflow && !iNoMoreSoundDeviceData)
502 //this sends a request to the hw device observer usually Devsound
503 //to update the bytes played
504 //it is done here so that the sound driver can be closed prior to
505 //updating the plicy and sending the error back
506 TUid uidUpdateBytesPlayed;
507 uidUpdateBytesPlayed.iUid = KMmfHwDeviceObserverUpdateBytesPlayed;
509 iHwDeviceObserver->MsgFromHwDevice(uidUpdateBytesPlayed,dummy);
511 //this closes RMdaDevSound.
514 //inform devsound so it can update policy
515 iHwDeviceObserver->Stopped();
517 // Inform the observer of the exception condition
518 // We inform the hw device observer after the policy has been
519 // updated incase the observer relied on the error to assume
520 // the policy has been updated
521 iHwDeviceObserver->Error(aError);
525 void CMMFSwCodecPlayDataPath::SetPlayCustomInterface(MPlayCustomInterface& aCustomInterface)
527 iCustomInterface = &aCustomInterface;
531 Retrieves a custom interface to the device.
532 The reference CMMFSwCodecWrapper supports two custom interfaces,
533 MEmptyBuffersCustomInterface and MSetVbrFlagCustomInterface
536 Interface UID, defined with the custom interface.
537 aInterface = KMmfUidEmptyBuffersCustomInterface for MEmptyBuffersCustomInterface,
538 KSetVbrFlagCustomInterfaceTypeUid for MSetVbrFlagCustomInterface
540 @return A pointer to the interface implementation, or NULL if the device can not
541 implement the interface requested. The return value must be cast to the
542 correct type by the user.
544 TAny* CMMFSwCodecPlayDataPath::CustomInterface(TUid aInterface)
547 if (aInterface.iUid == KMmfUidEmptyBuffersCustomInterface)
549 MEmptyBuffersCustomInterface* result = static_cast<MEmptyBuffersCustomInterface*> (this);
550 ret = static_cast<TAny*>(result);
552 else if(aInterface.iUid == KSetVbrFlagCustomInterfaceTypeUid)
556 if (aInterface == KTimePlayedCustomInterfaceTypeUid)
558 MTimePlayedCustomInterface* result = static_cast<MTimePlayedCustomInterface*> (this);
559 ret = static_cast<TAny*>(result);
561 if (aInterface == KIgnoreUnderflowCustomInterfaceTypeUid)
563 MIgnoreUnderflowEventsCustomInterface* result = static_cast<MIgnoreUnderflowEventsCustomInterface*> (this);
564 ret = static_cast<TAny*>(result);
570 Used to set iVbrFlag on the datapath.
572 This method is used to set the iVbrFlag in datapath. This flag is added to datapath to avail the
573 alternative dataflow wherein datapath makes sure that destinationbuffer is filled to its maximum length
574 before sending it to the sound driver. Sending the buffer directly to the device causes underflow incase of VBR codecs.
576 void CMMFSwCodecPlayDataPath::SetVbrFlag()
581 TInt CMMFSwCodecPlayDataPath::GetTimePlayed(TTimeIntervalMicroSeconds& aTime)
583 if(iSoundDevice.Handle())
585 TInt bytes = iSoundDevice.BytesPlayed();
586 //Work around for overflow of bytes played from driver.
587 //This code will be removed when Base provides the TimePlayed() API which returns the play time
588 //Assuming GetTimePlayed() gets called in an interval not more than 3 secs, reset driver's bytes played when it is near KMaxInt
589 if(bytes > (KMaxTInt - 3*KMaxBytesInSec))
591 iBytesPlayed = iBytesPlayed+bytes;
592 iSoundDevice.ResetBytesPlayed();
595 TInt64 samplesPlayed = (iBytesPlayed+bytes)/(KBytesPerSample*iChannels);
596 aTime = (samplesPlayed*1000000)/iSampleRate;
606 void CMMFSwCodecPlayDataPath::IgnoreUnderflowEvents()
608 iIgnoreUnderflow = ETrue;
610 /************************************************************************
612 ************************************************************************/
614 CDataPathPlayer::CDataPathPlayer(CMMFSwCodecPlayDataPath& aParent, TInt aPriority)
615 : CActive(aPriority), iParent(aParent)
617 CActiveScheduler::Add(this);
621 CDataPathPlayer::~CDataPathPlayer()
627 void CDataPathPlayer::Start()
633 void CDataPathPlayer::ResumePlaying()
635 if (iParent.Device().Handle())
637 //should be ok to call this even if we are active
638 iParent.Device().ResumePlaying();
639 iResumePlaying = ETrue;
642 RDebug::Print(_L("Playing Resumed"));
647 void CDataPathPlayer::PlayData(const CMMFDataBuffer& aData)
649 iDataFromSource = &aData;
653 RDebug::Print(_L("CDataPathPlayer::PlayData"));
655 iParent.Device().PlayData(iStatus,(STATIC_CAST(const CMMFDataBuffer*, iDataFromSource))->Data());
661 void CDataPathPlayer::Stop()
664 iParent.Device().FlushPlayBuffer(); // Otherwise won't be flushed
666 iParent.SoundDeviceException(KErrCancel);
670 void CDataPathPlayer::RunL()
673 RDebug::Print(_L("CDataPathPlayer::RunL error[%d]"), iStatus.Int());
675 if (iStatus.Int()!=KErrNone)
677 iParent.SoundDeviceException(iStatus.Int());
681 iParent.BufferEmptiedL(static_cast<const CMMFDataBuffer&>(*iDataFromSource));
682 iResumePlaying = EFalse;
687 TInt CDataPathPlayer::RunError(TInt aError)
694 void CDataPathPlayer::DoCancel()
696 if (iParent.Device().Handle())
698 iParent.Device().CancelPlayData();
699 iParent.Device().FlushPlayBuffer();
704 void CDataPathPlayer::Error(TInt aError)
706 iParent.SoundDeviceException(aError);
710 /************************************************************************
711 * CSoundDevPlayErrorReceiver *
712 ************************************************************************/
714 CSoundDevPlayErrorReceiver::CSoundDevPlayErrorReceiver(CMMFSwCodecPlayDataPath& aParent, TInt aPriority)
715 : CActive(aPriority), iParent(aParent)
717 CActiveScheduler::Add(this);
720 CSoundDevPlayErrorReceiver::~CSoundDevPlayErrorReceiver()
725 void CSoundDevPlayErrorReceiver::Start()
727 iParent.Device().NotifyPlayError(iStatus);
731 void CSoundDevPlayErrorReceiver::Stop()
736 void CSoundDevPlayErrorReceiver::RunL()
738 TInt reason = iStatus.Int();
740 // An error has been returned
742 RDebug::Print(_L("CSoundDevPlayErrorReceiver::RunL[%d]"), reason);
744 iParent.SoundDeviceException(reason);
747 void CSoundDevPlayErrorReceiver::DoCancel()
749 iParent.Device().CancelNotifyPlayError();