1.1 --- /dev/null Thu Jan 01 00:00:00 1970 +0000
1.2 +++ b/os/mm/devsound/devsoundrefplugin/src/swcodecwrapper/mmfSwCodecPlayDataPath.cpp Fri Jun 15 03:10:57 2012 +0200
1.3 @@ -0,0 +1,752 @@
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 +// source\server\mmfswcodecplaydatapath.cpp
1.18 +//
1.19 +//
1.20 +
1.21 +#include "mmfSwCodecPlayDataPath.h"
1.22 +#include <mmf/server/mmfswcodecwrapper.h>
1.23 +#include <mmf/server/mmfswcodecwrappercustominterfacesuids.hrh>
1.24 +#include <mmf/common/mmfpaniccodes.h>
1.25 +#include "mmfSwCodecUtility.h"
1.26 +
1.27 +const TInt KBytesPerSample = 2;
1.28 +const TInt KMaxBytesInSec = 192000; //considering maximum samplerate 96KHz
1.29 +CMMFSwCodecPlayDataPath* CMMFSwCodecPlayDataPath::NewL()
1.30 + {
1.31 + CMMFSwCodecPlayDataPath* self = new(ELeave) CMMFSwCodecPlayDataPath;
1.32 + CleanupStack::PushL(self);
1.33 + self->ConstructL();
1.34 + CleanupStack::Pop();
1.35 + return self;
1.36 + }
1.37 +
1.38 +
1.39 +void CMMFSwCodecPlayDataPath::ConstructL()
1.40 + {
1.41 + iAudioPlayer = new (ELeave) CDataPathPlayer(*this,CActive::EPriorityUserInput);
1.42 + iSoundDeviceErrorReceiver = new (ELeave) CSoundDevPlayErrorReceiver(*this, CActive::EPriorityUserInput);
1.43 + iUtility = CMMFSwCodecUtility::NewL();
1.44 + iVbrFlag = EFalse;
1.45 + }
1.46 +
1.47 +
1.48 +CMMFSwCodecPlayDataPath::~CMMFSwCodecPlayDataPath()
1.49 + {
1.50 + delete iAudioPlayer;
1.51 + delete iSoundDeviceErrorReceiver;
1.52 + delete iUtility;
1.53 +
1.54 + iSoundDevice.Close();
1.55 +
1.56 + if (iCodec)
1.57 + {
1.58 + delete iSourceBuffer;
1.59 + if (!iCodec->IsNullCodec())
1.60 + {
1.61 + delete iSoundDeviceBuffer;
1.62 + }
1.63 + }
1.64 +
1.65 +#ifdef __USE_MMF_TRANSFERBUFFERS__
1.66 + delete iTransferWindow;
1.67 +
1.68 + if(iTransferBuffer)
1.69 + {
1.70 + iTransferBuffer->Close();
1.71 + delete iTransferBuffer;
1.72 + }
1.73 +#endif
1.74 +
1.75 +#ifdef __USE_MMF_PTRBUFFERS__
1.76 + delete iPtrBufferMemoryBlock;
1.77 +#endif
1.78 + }
1.79 +
1.80 +
1.81 +TInt CMMFSwCodecPlayDataPath::SetObserver(MMMFHwDeviceObserver& aObserver)
1.82 + {
1.83 + TInt error;
1.84 + if (iHwDeviceObserver)
1.85 + {
1.86 + error = KErrAlreadyExists;
1.87 + }
1.88 + else
1.89 + {
1.90 + iHwDeviceObserver = &aObserver;
1.91 + error = KErrNone;
1.92 + }
1.93 + return error;
1.94 + }
1.95 +
1.96 +
1.97 +TInt CMMFSwCodecPlayDataPath::AddCodec(CMMFSwCodec& aCodec)
1.98 + {
1.99 + if (iCodec)
1.100 + return KErrNotSupported; //doesn't support multiple codecs
1.101 +
1.102 + TInt err = KErrNone;
1.103 +
1.104 + iCodec = &aCodec;
1.105 +
1.106 + // Allocate data buffer
1.107 + iSourceBufferSize = iCodec->SourceBufferSize();
1.108 + iSoundDevBufferSize = iCodec->SinkBufferSize();
1.109 +
1.110 + if ((!iSourceBufferSize)||(!iSoundDevBufferSize))
1.111 + err = KErrArgument; //codec plugin has not specified buffer size
1.112 +
1.113 + if (err == KErrNone)
1.114 + {
1.115 +#ifdef __USE_MMF_TRANSFERBUFFERS__
1.116 + TRAP(err,iSourceBuffer = CreateTransferBufferL(iSourceBufferSize, static_cast<CMMFTransferBuffer*>(iSourceBuffer)));
1.117 +#endif
1.118 +#ifdef __USE_MMF_PTRBUFFERS__
1.119 + TRAP(err,iSourceBuffer = CreatePtrBufferL(iSourceBufferSize));
1.120 +#else
1.121 + TRAP(err,iSourceBuffer = CMMFDataBuffer::NewL(iSourceBufferSize));
1.122 +#endif
1.123 + }
1.124 +
1.125 + if (err == KErrNone)
1.126 + {
1.127 + if (iCodec->IsNullCodec())
1.128 + {//use source buffer for sound device buffer
1.129 + iSoundDeviceBuffer = NULL;
1.130 + }
1.131 + else
1.132 + {//codec needs separate source and sound device buffers
1.133 + TRAP(err,iSoundDeviceBuffer = CMMFDataBuffer::NewL(iSoundDevBufferSize));
1.134 + }
1.135 + }
1.136 + return err;
1.137 + }
1.138 +
1.139 +
1.140 +TInt CMMFSwCodecPlayDataPath::Start()
1.141 + {
1.142 + TInt startError = KErrNone;
1.143 +
1.144 + if (!iCodec)
1.145 + {//check that a codec has been added
1.146 + startError = KErrNotReady;
1.147 + }
1.148 + if ((!iSoundDevice.Handle())&&(!startError))
1.149 + {//check that the sound drivers can be opened
1.150 + startError = iSoundDevice.Open();
1.151 + }
1.152 +
1.153 + if (iState == EPaused)
1.154 + {//we are paused so need to resume play
1.155 + if (!startError)
1.156 + {
1.157 +#ifdef _SCW_DEBUG
1.158 + RDebug::Print(_L("CMMFSwCodecPlayDataPath::Start-Resume"));
1.159 +#endif
1.160 + iAudioPlayer->ResumePlaying();
1.161 + iState = EPlaying;
1.162 + }
1.163 + }
1.164 + else if (!startError)
1.165 + {
1.166 +#ifdef _SCW_DEBUG
1.167 + RDebug::Print(_L("CMMFSwCodecPlayDataPath::Start-Normal"));
1.168 +#endif
1.169 + // get sample rate and channels from RMdaDevSound
1.170 + RMdaDevSound::TCurrentSoundFormatBuf format;
1.171 + iSoundDevice.GetPlayFormat(format);
1.172 + iSampleRate = format().iRate;
1.173 + iChannels = format().iChannels;
1.174 +
1.175 + iNoMoreSourceData = EFalse;
1.176 + iNoMoreSoundDeviceData = EFalse;
1.177 + iSourceBuffer->SetLastBuffer(EFalse);
1.178 + iBytesPlayed = 0;
1.179 + iState = EPlaying;
1.180 + iSoundDeviceErrorReceiver->Start();
1.181 + TRAP(startError,FillSourceBufferL()); //get initial buffer of audio data
1.182 + if (startError == KErrNone)
1.183 + {
1.184 + // Start the player objects
1.185 + iAudioPlayer->Start();
1.186 + }
1.187 + else
1.188 + {//failed to start up correctly go back to stopped state
1.189 + iState = EStopped;
1.190 + iSoundDeviceErrorReceiver->Stop();
1.191 + }
1.192 + }
1.193 + return startError;
1.194 + }
1.195 +
1.196 +
1.197 +// *** Main Play Loop ***
1.198 +
1.199 +void CMMFSwCodecPlayDataPath::FillSourceBufferL()
1.200 + {//asks observer to fill the source buffer
1.201 + // Ask immediately for data from the observer
1.202 +#ifdef __CYCLE_MMF_DATABUFFERS__
1.203 + // Create a new buffer to replicate INC021405 Play-EOF-Play on HwAccelerated solution Panics
1.204 + // If the creation fails, we carry on regardless as the original buffer will not have been
1.205 + // destroyed. Must do this as alloc fail tests will not run.
1.206 + if(iSourceBuffer)
1.207 + {
1.208 + iSourceBuffer = CycleAudioBuffer(iSourceBuffer);
1.209 + }
1.210 +#endif // __CYCLE_MMF_DATABUFFERS__
1.211 + User::LeaveIfError(iHwDeviceObserver->FillThisHwBuffer(*iSourceBuffer));
1.212 +
1.213 + }
1.214 +
1.215 +
1.216 +void CMMFSwCodecPlayDataPath::BufferFilledL(CMMFDataBuffer& aBuffer)
1.217 + {//call back from observer to indicate buffer has been filled
1.218 + if (iState == EStopped)
1.219 + User::Leave(KErrNotReady);//ok if paused?
1.220 +
1.221 + iSourceBuffer = &aBuffer;
1.222 + iSourceBuffer->SetStatus(EFull);
1.223 +#ifdef _SCW_DEBUG
1.224 + RDebug::Print(_L("CMMFSwCodecPlayDataPath::BufferFilledL"));
1.225 +#endif
1.226 +
1.227 + //need to check that the buffer size is not 0 - if so assume we've reached the end of the data
1.228 + if (!iSourceBuffer->BufferSize())
1.229 + {//no buffer - could be end of source or could be that the source has no data??
1.230 + iNoMoreSourceData = ETrue;
1.231 +#ifdef _SCW_DEBUG
1.232 + RDebug::Print(_L("CMMFSwCodecPlayDataPath::BufferFilledL-NoMoreSourceData"));
1.233 +#endif
1.234 + }
1.235 + //even if the buffer size is 0 we still
1.236 + //need to perform the following to get the sound device callback
1.237 + FillSoundDeviceBufferL(); //get buffer in pcm16 format for sound device
1.238 +
1.239 + /* iVbrFlag is added to datapath to avail the alternative dataflow wherein datapath makes sure that
1.240 + destinationbuffer is filled to its maximum length before sending it to the sound driver.
1.241 + Sending the buffer directly to the device causes underflow incase of Vorbis codecs.*/
1.242 + if (iVbrFlag)
1.243 + {
1.244 + /*There are two cases we need to deal here
1.245 + 1. When the output of the codec is 0 for header data.
1.246 + in that case, repeat till actual decoding of ogg packets and pages.
1.247 + 2. When destination buffer is not filled even to its half length, get next source buffer
1.248 + and decode it. This is to avoid underflows when ever we receive little pcm for a
1.249 + a given source buffer.
1.250 + */
1.251 + if (iSoundDeviceBuffer->Data().Length() < iSoundDeviceBuffer->Data().MaxLength()/2 && !(iSoundDeviceBuffer->LastBuffer()))
1.252 + {
1.253 + iSourceBuffer->SetStatus(EAvailable); //source buffer is now available
1.254 + iSoundDeviceBuffer->SetPosition(iSoundDeviceBuffer->Data().Length());//this indicates the available space in the buffer to the codec
1.255 + FillSourceBufferL();
1.256 + return;
1.257 + }
1.258 + else //data is sufficient to avoid underflows
1.259 + {
1.260 + iSoundDeviceBuffer->SetPosition(0);
1.261 + if(iSoundDeviceBuffer->Data().Length()==0 && iSoundDeviceBuffer->LastBuffer())
1.262 + {
1.263 + iNoMoreSoundDeviceData = ETrue;
1.264 + }
1.265 + }
1.266 + }
1.267 +
1.268 + // attenuate the amplitude of the samples if volume ramping has been changed
1.269 + // and is non-zero
1.270 + if (iCustomInterface)
1.271 + {
1.272 + TTimeIntervalMicroSeconds volumeRamp = iCustomInterface->VolumeRamp();
1.273 + if (volumeRamp != iVolumeRamp)
1.274 + {
1.275 + iVolumeRamp = volumeRamp;
1.276 + if (iVolumeRamp.Int64() != 0)
1.277 + {
1.278 + iUtility->ConfigAudioRamper(
1.279 + iVolumeRamp.Int64(),
1.280 + iSampleRate,
1.281 + iChannels);
1.282 + iRampAudioSample = ETrue;
1.283 + }
1.284 + else
1.285 + {
1.286 + iRampAudioSample = EFalse;
1.287 + }
1.288 +
1.289 + }
1.290 + if (iRampAudioSample)
1.291 + iRampAudioSample = iUtility->RampAudio(iSoundDeviceBuffer);
1.292 + }
1.293 +
1.294 + iAudioPlayer->PlayData(*iSoundDeviceBuffer); //play data to sound drivers
1.295 +
1.296 + if (iSourceBuffer->LastBuffer())//check last buffer flag
1.297 + {
1.298 + iNoMoreSourceData = ETrue;
1.299 +#ifdef _SCW_DEBUG
1.300 + RDebug::Print(_L("CMMFSwCodecPlayDataPath::BufferFilledL-LBNoMoreSourceData"));
1.301 +#endif
1.302 + }
1.303 + }
1.304 +
1.305 +
1.306 +void CMMFSwCodecPlayDataPath::FillSoundDeviceBufferL()
1.307 + {//use CMMFSwCodec to fill the sound device buffer
1.308 +
1.309 + CMMFSwCodec::TCodecProcessResult codecProcessResult;
1.310 +
1.311 + if (iCodec->IsNullCodec())
1.312 + {//no codec so data can be sent direct to sink
1.313 + iSoundDeviceBuffer = iSourceBuffer;
1.314 + iSoundDeviceBuffer->SetStatus(EFull); //sink buffer is full
1.315 + }
1.316 + else
1.317 + {
1.318 + //pass buffer to codec for processing
1.319 + codecProcessResult = iCodec->ProcessL(*iSourceBuffer, *iSoundDeviceBuffer);
1.320 +
1.321 + if (iSourceBuffer->LastBuffer()) //if source is last buffer so is sound dev
1.322 + iSoundDeviceBuffer->SetLastBuffer(ETrue);
1.323 + if ((!iSoundDeviceBuffer->BufferSize())&&(codecProcessResult.iDstBytesAdded))
1.324 + {//the codec has added data but not set the buffer length
1.325 + iSoundDeviceBuffer->Data().SetLength(codecProcessResult.iDstBytesAdded);
1.326 + }
1.327 + //only supports EProcessComplete
1.328 + switch (codecProcessResult.iCodecProcessStatus)
1.329 + {
1.330 + case CMMFSwCodec::TCodecProcessResult::EProcessComplete:
1.331 + //finished procesing source data - all data in sink buffer
1.332 + {
1.333 + iSoundDeviceBuffer->SetStatus(EFull); //sink buffer is full
1.334 + }
1.335 + break;
1.336 +#ifdef SYMBIAN_VARIABLE_BITRATE_CODEC
1.337 + case CMMFSwCodec::TCodecProcessResult::EProcessIncomplete:
1.338 + //finished procesing source data - all data in sink buffer
1.339 + {
1.340 + iSoundDeviceBuffer->SetStatus(EFull); //sink buffer is full
1.341 + }
1.342 + break;
1.343 +#endif
1.344 + case CMMFSwCodec::TCodecProcessResult::EDstNotFilled:
1.345 + //could be the last buffer in which case dst might not get filled
1.346 + {
1.347 + iSoundDeviceBuffer->SetStatus(EFull); //sink buffer is full
1.348 + }
1.349 + break;
1.350 + case CMMFSwCodec::TCodecProcessResult::EEndOfData:
1.351 + //no more data - send what we've got to the sink
1.352 + //note we can't always rely on this - in many cases the codec will not know when
1.353 + //it has reached the end of data.
1.354 + {
1.355 + iSoundDeviceBuffer->SetStatus(EFull);//sink buffer may not really be 'full' but its as full as it going to get
1.356 + iNoMoreSourceData = ETrue;
1.357 + //doesn't matter if sink buffer is not full
1.358 + }
1.359 + break;
1.360 + default:
1.361 + Panic(EMMFSwCodecWrapperBadCodec); //should never get here - bad codec
1.362 + }
1.363 + }
1.364 + }
1.365 +
1.366 +
1.367 +void CMMFSwCodecPlayDataPath::BufferEmptiedL(const CMMFDataBuffer& aBuffer)
1.368 + {//call back from CDataPathPlayer when the sound device buffer has been emptied
1.369 + if (&aBuffer != iSoundDeviceBuffer)
1.370 + Panic(EMMFSwCodecWrapperBadBuffer);
1.371 + if(iVbrFlag && (iSourceBuffer->Status() == EUnAvailable || iNoMoreSourceData))
1.372 + {//No more source data. Play rest of the decoded data.Inform codec not to consider the source buffer
1.373 + if(iSourceBuffer->Status()!=EUnAvailable)
1.374 + {
1.375 + iSourceBuffer->SetStatus(EUnAvailable);
1.376 + }
1.377 + FillSoundDeviceBufferL();
1.378 + if(iSoundDeviceBuffer->BufferSize() > 0)
1.379 + {
1.380 + // attenuate the amplitude of the samples if volume ramping has been changed
1.381 + // and is non-zero
1.382 + if (iCustomInterface)
1.383 + {
1.384 + TTimeIntervalMicroSeconds volumeRamp = iCustomInterface->VolumeRamp();
1.385 + if (volumeRamp != iVolumeRamp)
1.386 + {
1.387 + iVolumeRamp = volumeRamp;
1.388 + if (iVolumeRamp.Int64() != 0)
1.389 + {
1.390 + iUtility->ConfigAudioRamper(iVolumeRamp.Int64(), iSampleRate, iChannels);
1.391 + iRampAudioSample = ETrue;
1.392 + }
1.393 + else
1.394 + {
1.395 + iRampAudioSample = EFalse;
1.396 + }
1.397 +
1.398 + }
1.399 + if (iRampAudioSample)
1.400 + {
1.401 + iRampAudioSample = iUtility->RampAudio(iSoundDeviceBuffer);
1.402 + }
1.403 +
1.404 + }
1.405 + iAudioPlayer->PlayData(*iSoundDeviceBuffer); //play data to sound drivers
1.406 + return;
1.407 + }
1.408 + else
1.409 + {
1.410 + if(iNoMoreSourceData)
1.411 + {
1.412 + iNoMoreSoundDeviceData = ETrue;
1.413 + }
1.414 + iSourceBuffer->SetStatus(EAvailable);
1.415 + }
1.416 + }
1.417 + if (!iNoMoreSourceData)
1.418 + FillSourceBufferL();
1.419 + }
1.420 +
1.421 +//*** End of Main Play Loop ***
1.422 +
1.423 +
1.424 +void CMMFSwCodecPlayDataPath::Stop()
1.425 + {
1.426 + iAudioPlayer->Cancel();
1.427 + iSoundDeviceErrorReceiver->Cancel();
1.428 + iSoundDevice.Close();
1.429 +
1.430 +#ifdef __CYCLE_MMF_DATABUFFERS__
1.431 + // Create a new buffer to replicate INC021405 Play-EOF-Play on HwAccelerated solution Panics
1.432 + // If the creation fails, we carry on regardless as the original buffer will not have been
1.433 + // destroyed. Must do this as alloc fail tests will not run.
1.434 + if(iSourceBuffer)
1.435 + {
1.436 + iSourceBuffer = CycleAudioBuffer(iSourceBuffer);
1.437 + }
1.438 +#endif // __CYCLE_MMF_DATABUFFERS__
1.439 +
1.440 + iState = EStopped;
1.441 + }
1.442 +
1.443 +
1.444 +void CMMFSwCodecPlayDataPath::Pause()
1.445 + {
1.446 + //since a pause can happen anyway in the datatransfer -need to set to a known
1.447 + //state so that when play is resumed the behaviour is predictable
1.448 + if (iSoundDevice.Handle())
1.449 + {
1.450 + iSoundDevice.PausePlayBuffer(); //needs new LDD
1.451 + iState = EPaused;
1.452 +#ifdef _SCW_DEBUG
1.453 + RDebug::Print(_L("Pause"));
1.454 +#endif
1.455 + }
1.456 + else
1.457 + {//an error must have occured
1.458 + iState = EStopped;
1.459 + }
1.460 + }
1.461 +
1.462 +
1.463 +TInt CMMFSwCodecPlayDataPath::EmptyBuffers()
1.464 + {
1.465 + TInt error = KErrNone;
1.466 + if (iSoundDevice.Handle() == 0)
1.467 + {
1.468 + error = KErrNotReady;
1.469 + }
1.470 + else
1.471 + {
1.472 + iAudioPlayer->Cancel();
1.473 + iSoundDevice.CancelPlayData();
1.474 + iSoundDeviceErrorReceiver->Stop();
1.475 + iState = EStopped;
1.476 + }
1.477 + return error;
1.478 + }
1.479 +
1.480 +
1.481 +RMdaDevSound& CMMFSwCodecPlayDataPath::Device()
1.482 + {
1.483 + return iSoundDevice;
1.484 + }
1.485 +
1.486 +
1.487 +void CMMFSwCodecPlayDataPath::SoundDeviceException(TInt aError)
1.488 + {
1.489 + if(iIgnoreUnderflow)
1.490 + {
1.491 + if(!iVbrFlag && aError==KErrUnderflow && !iNoMoreSourceData)
1.492 + {
1.493 + //ignore underflow
1.494 + return;
1.495 + }
1.496 + //for VBR codec data,no more source does not mean that no more sounddevice data
1.497 + //so ignore underflows till the last buffer is played from the codec
1.498 + else if(iVbrFlag && aError==KErrUnderflow && !iNoMoreSoundDeviceData)
1.499 + {
1.500 + //ignore underflow
1.501 + return;
1.502 + }
1.503 + }
1.504 +
1.505 + //this sends a request to the hw device observer usually Devsound
1.506 + //to update the bytes played
1.507 + //it is done here so that the sound driver can be closed prior to
1.508 + //updating the plicy and sending the error back
1.509 + TUid uidUpdateBytesPlayed;
1.510 + uidUpdateBytesPlayed.iUid = KMmfHwDeviceObserverUpdateBytesPlayed;
1.511 + TPtrC8 dummy(0,0);
1.512 + iHwDeviceObserver->MsgFromHwDevice(uidUpdateBytesPlayed,dummy);
1.513 +
1.514 + //this closes RMdaDevSound.
1.515 + Stop();
1.516 +
1.517 + //inform devsound so it can update policy
1.518 + iHwDeviceObserver->Stopped();
1.519 +
1.520 + // Inform the observer of the exception condition
1.521 + // We inform the hw device observer after the policy has been
1.522 + // updated incase the observer relied on the error to assume
1.523 + // the policy has been updated
1.524 + iHwDeviceObserver->Error(aError);
1.525 + }
1.526 +
1.527 +
1.528 +void CMMFSwCodecPlayDataPath::SetPlayCustomInterface(MPlayCustomInterface& aCustomInterface)
1.529 + {
1.530 + iCustomInterface = &aCustomInterface;
1.531 + }
1.532 +
1.533 +/**
1.534 +Retrieves a custom interface to the device.
1.535 +The reference CMMFSwCodecWrapper supports two custom interfaces,
1.536 +MEmptyBuffersCustomInterface and MSetVbrFlagCustomInterface
1.537 +
1.538 +@param aInterface
1.539 + Interface UID, defined with the custom interface.
1.540 + aInterface = KMmfUidEmptyBuffersCustomInterface for MEmptyBuffersCustomInterface,
1.541 + KSetVbrFlagCustomInterfaceTypeUid for MSetVbrFlagCustomInterface
1.542 +
1.543 +@return A pointer to the interface implementation, or NULL if the device can not
1.544 + implement the interface requested. The return value must be cast to the
1.545 + correct type by the user.
1.546 +*/
1.547 +TAny* CMMFSwCodecPlayDataPath::CustomInterface(TUid aInterface)
1.548 + {
1.549 + TAny* ret = NULL;
1.550 + if (aInterface.iUid == KMmfUidEmptyBuffersCustomInterface)
1.551 + {
1.552 + MEmptyBuffersCustomInterface* result = static_cast<MEmptyBuffersCustomInterface*> (this);
1.553 + ret = static_cast<TAny*>(result);
1.554 + }
1.555 + else if(aInterface.iUid == KSetVbrFlagCustomInterfaceTypeUid)
1.556 + {
1.557 + SetVbrFlag();
1.558 + }
1.559 + if (aInterface == KTimePlayedCustomInterfaceTypeUid)
1.560 + {
1.561 + MTimePlayedCustomInterface* result = static_cast<MTimePlayedCustomInterface*> (this);
1.562 + ret = static_cast<TAny*>(result);
1.563 + }
1.564 + if (aInterface == KIgnoreUnderflowCustomInterfaceTypeUid)
1.565 + {
1.566 + MIgnoreUnderflowEventsCustomInterface* result = static_cast<MIgnoreUnderflowEventsCustomInterface*> (this);
1.567 + ret = static_cast<TAny*>(result);
1.568 + }
1.569 + return ret;
1.570 + }
1.571 +
1.572 +/**
1.573 +Used to set iVbrFlag on the datapath.
1.574 +
1.575 +This method is used to set the iVbrFlag in datapath. This flag is added to datapath to avail the
1.576 +alternative dataflow wherein datapath makes sure that destinationbuffer is filled to its maximum length
1.577 +before sending it to the sound driver. Sending the buffer directly to the device causes underflow incase of VBR codecs.
1.578 +*/
1.579 +void CMMFSwCodecPlayDataPath::SetVbrFlag()
1.580 + {
1.581 + iVbrFlag = ETrue;
1.582 + }
1.583 +
1.584 +TInt CMMFSwCodecPlayDataPath::GetTimePlayed(TTimeIntervalMicroSeconds& aTime)
1.585 + {
1.586 + if(iSoundDevice.Handle())
1.587 + {
1.588 + TInt bytes = iSoundDevice.BytesPlayed();
1.589 + //Work around for overflow of bytes played from driver.
1.590 + //This code will be removed when Base provides the TimePlayed() API which returns the play time
1.591 + //Assuming GetTimePlayed() gets called in an interval not more than 3 secs, reset driver's bytes played when it is near KMaxInt
1.592 + if(bytes > (KMaxTInt - 3*KMaxBytesInSec))
1.593 + {
1.594 + iBytesPlayed = iBytesPlayed+bytes;
1.595 + iSoundDevice.ResetBytesPlayed();
1.596 + bytes = 0;
1.597 + }
1.598 + TInt64 samplesPlayed = (iBytesPlayed+bytes)/(KBytesPerSample*iChannels);
1.599 + aTime = (samplesPlayed*1000000)/iSampleRate;
1.600 + }
1.601 + else
1.602 + {
1.603 + aTime = 0;
1.604 + }
1.605 +
1.606 + return KErrNone;
1.607 + }
1.608 +
1.609 +void CMMFSwCodecPlayDataPath::IgnoreUnderflowEvents()
1.610 + {
1.611 + iIgnoreUnderflow = ETrue;
1.612 + }
1.613 +/************************************************************************
1.614 + * CDataPathPlayer *
1.615 + ************************************************************************/
1.616 +
1.617 +CDataPathPlayer::CDataPathPlayer(CMMFSwCodecPlayDataPath& aParent, TInt aPriority)
1.618 +: CActive(aPriority), iParent(aParent)
1.619 + {
1.620 + CActiveScheduler::Add(this);
1.621 + }
1.622 +
1.623 +
1.624 +CDataPathPlayer::~CDataPathPlayer()
1.625 + {
1.626 + Cancel();
1.627 + }
1.628 +
1.629 +
1.630 +void CDataPathPlayer::Start()
1.631 + {
1.632 + // No implementation
1.633 + }
1.634 +
1.635 +
1.636 +void CDataPathPlayer::ResumePlaying()
1.637 + {
1.638 + if (iParent.Device().Handle())
1.639 + {
1.640 + //should be ok to call this even if we are active
1.641 + iParent.Device().ResumePlaying();
1.642 + iResumePlaying = ETrue;
1.643 + }
1.644 +#ifdef _SCW_DEBUG
1.645 + RDebug::Print(_L("Playing Resumed"));
1.646 +#endif
1.647 + }
1.648 +
1.649 +
1.650 +void CDataPathPlayer::PlayData(const CMMFDataBuffer& aData)
1.651 + {
1.652 + iDataFromSource = &aData;
1.653 + if (!IsActive())
1.654 + {
1.655 +#ifdef _SCW_DEBUG
1.656 + RDebug::Print(_L("CDataPathPlayer::PlayData"));
1.657 +#endif
1.658 + iParent.Device().PlayData(iStatus,(STATIC_CAST(const CMMFDataBuffer*, iDataFromSource))->Data());
1.659 + SetActive();
1.660 + }
1.661 + }
1.662 +
1.663 +
1.664 +void CDataPathPlayer::Stop()
1.665 + {
1.666 + if (!IsActive())
1.667 + iParent.Device().FlushPlayBuffer(); // Otherwise won't be flushed
1.668 + Cancel();
1.669 + iParent.SoundDeviceException(KErrCancel);
1.670 + }
1.671 +
1.672 +
1.673 +void CDataPathPlayer::RunL()
1.674 + {
1.675 +#ifdef _SCW_DEBUG
1.676 + RDebug::Print(_L("CDataPathPlayer::RunL error[%d]"), iStatus.Int());
1.677 +#endif
1.678 + if (iStatus.Int()!=KErrNone)
1.679 + {
1.680 + iParent.SoundDeviceException(iStatus.Int());
1.681 + }
1.682 + else
1.683 + {
1.684 + iParent.BufferEmptiedL(static_cast<const CMMFDataBuffer&>(*iDataFromSource));
1.685 + iResumePlaying = EFalse;
1.686 + }
1.687 + }
1.688 +
1.689 +
1.690 +TInt CDataPathPlayer::RunError(TInt aError)
1.691 + {
1.692 + Error(aError);
1.693 + return KErrNone;
1.694 + }
1.695 +
1.696 +
1.697 +void CDataPathPlayer::DoCancel()
1.698 + {
1.699 + if (iParent.Device().Handle())
1.700 + {
1.701 + iParent.Device().CancelPlayData();
1.702 + iParent.Device().FlushPlayBuffer();
1.703 + }
1.704 + }
1.705 +
1.706 +
1.707 +void CDataPathPlayer::Error(TInt aError)
1.708 + {
1.709 + iParent.SoundDeviceException(aError);
1.710 + }
1.711 +
1.712 +
1.713 +/************************************************************************
1.714 + * CSoundDevPlayErrorReceiver *
1.715 + ************************************************************************/
1.716 +
1.717 +CSoundDevPlayErrorReceiver::CSoundDevPlayErrorReceiver(CMMFSwCodecPlayDataPath& aParent, TInt aPriority)
1.718 +: CActive(aPriority), iParent(aParent)
1.719 + {
1.720 + CActiveScheduler::Add(this);
1.721 + }
1.722 +
1.723 +CSoundDevPlayErrorReceiver::~CSoundDevPlayErrorReceiver()
1.724 + {
1.725 + Cancel();
1.726 + }
1.727 +
1.728 +void CSoundDevPlayErrorReceiver::Start()
1.729 + {
1.730 + iParent.Device().NotifyPlayError(iStatus);
1.731 + SetActive();
1.732 + }
1.733 +
1.734 +void CSoundDevPlayErrorReceiver::Stop()
1.735 + {
1.736 + Cancel();
1.737 + }
1.738 +
1.739 +void CSoundDevPlayErrorReceiver::RunL()
1.740 + {
1.741 + TInt reason = iStatus.Int();
1.742 + Start();
1.743 + // An error has been returned
1.744 +#ifdef _SCW_DEBUG
1.745 + RDebug::Print(_L("CSoundDevPlayErrorReceiver::RunL[%d]"), reason);
1.746 +#endif
1.747 + iParent.SoundDeviceException(reason);
1.748 + }
1.749 +
1.750 +void CSoundDevPlayErrorReceiver::DoCancel()
1.751 + {
1.752 + iParent.Device().CancelNotifyPlayError();
1.753 + }
1.754 +
1.755 +