os/mm/devsound/devsoundrefplugin/src/swcodecwrapper/mmfSwCodecPlayDataPath.cpp
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
Update contrib.
sl@0
     1
// Copyright (c) 2003-2009 Nokia Corporation and/or its subsidiary(-ies).
sl@0
     2
// All rights reserved.
sl@0
     3
// This component and the accompanying materials are made available
sl@0
     4
// under the terms of "Eclipse Public License v1.0"
sl@0
     5
// which accompanies this distribution, and is available
sl@0
     6
// at the URL "http://www.eclipse.org/legal/epl-v10.html".
sl@0
     7
//
sl@0
     8
// Initial Contributors:
sl@0
     9
// Nokia Corporation - initial contribution.
sl@0
    10
//
sl@0
    11
// Contributors:
sl@0
    12
//
sl@0
    13
// Description:
sl@0
    14
// source\server\mmfswcodecplaydatapath.cpp
sl@0
    15
// 
sl@0
    16
//
sl@0
    17
sl@0
    18
#include "mmfSwCodecPlayDataPath.h"
sl@0
    19
#include <mmf/server/mmfswcodecwrapper.h>
sl@0
    20
#include <mmf/server/mmfswcodecwrappercustominterfacesuids.hrh>
sl@0
    21
#include <mmf/common/mmfpaniccodes.h>
sl@0
    22
#include "mmfSwCodecUtility.h"
sl@0
    23
sl@0
    24
const TInt KBytesPerSample = 2;
sl@0
    25
const TInt KMaxBytesInSec = 192000; //considering maximum samplerate 96KHz
sl@0
    26
CMMFSwCodecPlayDataPath* CMMFSwCodecPlayDataPath::NewL()
sl@0
    27
	{
sl@0
    28
	CMMFSwCodecPlayDataPath* self = new(ELeave) CMMFSwCodecPlayDataPath;
sl@0
    29
	CleanupStack::PushL(self);
sl@0
    30
	self->ConstructL();
sl@0
    31
	CleanupStack::Pop();
sl@0
    32
	return self;
sl@0
    33
	}
sl@0
    34
sl@0
    35
sl@0
    36
void CMMFSwCodecPlayDataPath::ConstructL()
sl@0
    37
	{
sl@0
    38
	iAudioPlayer = new (ELeave) CDataPathPlayer(*this,CActive::EPriorityUserInput);
sl@0
    39
	iSoundDeviceErrorReceiver = new (ELeave) CSoundDevPlayErrorReceiver(*this, CActive::EPriorityUserInput);
sl@0
    40
	iUtility = CMMFSwCodecUtility::NewL();
sl@0
    41
	iVbrFlag = EFalse;
sl@0
    42
	}
sl@0
    43
sl@0
    44
sl@0
    45
CMMFSwCodecPlayDataPath::~CMMFSwCodecPlayDataPath()
sl@0
    46
	{
sl@0
    47
	delete iAudioPlayer;
sl@0
    48
	delete iSoundDeviceErrorReceiver;
sl@0
    49
	delete iUtility;
sl@0
    50
sl@0
    51
	iSoundDevice.Close();
sl@0
    52
sl@0
    53
	if (iCodec)
sl@0
    54
		{
sl@0
    55
		delete iSourceBuffer;
sl@0
    56
		if (!iCodec->IsNullCodec()) 
sl@0
    57
			{
sl@0
    58
			delete iSoundDeviceBuffer;
sl@0
    59
			}
sl@0
    60
		}
sl@0
    61
sl@0
    62
#ifdef __USE_MMF_TRANSFERBUFFERS__
sl@0
    63
	delete iTransferWindow;
sl@0
    64
sl@0
    65
	if(iTransferBuffer)
sl@0
    66
		{
sl@0
    67
		iTransferBuffer->Close();
sl@0
    68
		delete iTransferBuffer;
sl@0
    69
		}
sl@0
    70
#endif
sl@0
    71
sl@0
    72
#ifdef __USE_MMF_PTRBUFFERS__
sl@0
    73
	delete iPtrBufferMemoryBlock;
sl@0
    74
#endif
sl@0
    75
	}
sl@0
    76
sl@0
    77
sl@0
    78
TInt CMMFSwCodecPlayDataPath::SetObserver(MMMFHwDeviceObserver& aObserver)
sl@0
    79
	{
sl@0
    80
	TInt error;
sl@0
    81
	if (iHwDeviceObserver)
sl@0
    82
		{
sl@0
    83
		error =  KErrAlreadyExists;
sl@0
    84
		}
sl@0
    85
	else
sl@0
    86
		{
sl@0
    87
		iHwDeviceObserver = &aObserver;
sl@0
    88
		error  = KErrNone;
sl@0
    89
		}
sl@0
    90
	return error;
sl@0
    91
	}
sl@0
    92
sl@0
    93
sl@0
    94
TInt CMMFSwCodecPlayDataPath::AddCodec(CMMFSwCodec& aCodec)
sl@0
    95
	{
sl@0
    96
	if (iCodec)
sl@0
    97
		return KErrNotSupported; //doesn't support multiple codecs
sl@0
    98
sl@0
    99
	TInt err = KErrNone;
sl@0
   100
	
sl@0
   101
	iCodec = &aCodec;
sl@0
   102
sl@0
   103
	// Allocate data buffer
sl@0
   104
	iSourceBufferSize = iCodec->SourceBufferSize();
sl@0
   105
	iSoundDevBufferSize = iCodec->SinkBufferSize();
sl@0
   106
sl@0
   107
	if ((!iSourceBufferSize)||(!iSoundDevBufferSize))
sl@0
   108
		err = KErrArgument; //codec plugin has not specified buffer size
sl@0
   109
sl@0
   110
	if (err == KErrNone)
sl@0
   111
		{
sl@0
   112
#ifdef __USE_MMF_TRANSFERBUFFERS__
sl@0
   113
		TRAP(err,iSourceBuffer = CreateTransferBufferL(iSourceBufferSize, static_cast<CMMFTransferBuffer*>(iSourceBuffer)));
sl@0
   114
#endif
sl@0
   115
#ifdef __USE_MMF_PTRBUFFERS__
sl@0
   116
		TRAP(err,iSourceBuffer = CreatePtrBufferL(iSourceBufferSize));
sl@0
   117
#else
sl@0
   118
		TRAP(err,iSourceBuffer = CMMFDataBuffer::NewL(iSourceBufferSize));
sl@0
   119
#endif
sl@0
   120
		}
sl@0
   121
	
sl@0
   122
	if (err == KErrNone)
sl@0
   123
		{
sl@0
   124
		if (iCodec->IsNullCodec())
sl@0
   125
			{//use source buffer for sound device buffer	
sl@0
   126
			iSoundDeviceBuffer = NULL;
sl@0
   127
			}
sl@0
   128
		else
sl@0
   129
			{//codec needs separate source and sound device buffers
sl@0
   130
			TRAP(err,iSoundDeviceBuffer = CMMFDataBuffer::NewL(iSoundDevBufferSize));
sl@0
   131
			}
sl@0
   132
		}
sl@0
   133
	return err;
sl@0
   134
	}
sl@0
   135
sl@0
   136
sl@0
   137
TInt CMMFSwCodecPlayDataPath::Start()
sl@0
   138
	{
sl@0
   139
	TInt startError = KErrNone;
sl@0
   140
sl@0
   141
	if (!iCodec) 
sl@0
   142
		{//check that a codec has been added
sl@0
   143
		startError = KErrNotReady;
sl@0
   144
		}
sl@0
   145
	if ((!iSoundDevice.Handle())&&(!startError))
sl@0
   146
    	{//check that the sound drivers can be opened
sl@0
   147
   		startError = iSoundDevice.Open();
sl@0
   148
		}
sl@0
   149
sl@0
   150
	if (iState == EPaused)
sl@0
   151
		{//we are paused so need to resume play
sl@0
   152
		if (!startError)
sl@0
   153
			{
sl@0
   154
#ifdef _SCW_DEBUG
sl@0
   155
			RDebug::Print(_L("CMMFSwCodecPlayDataPath::Start-Resume"));
sl@0
   156
#endif
sl@0
   157
			iAudioPlayer->ResumePlaying();
sl@0
   158
			iState = EPlaying;
sl@0
   159
			}
sl@0
   160
		}
sl@0
   161
	else if (!startError)
sl@0
   162
		{
sl@0
   163
#ifdef _SCW_DEBUG
sl@0
   164
		RDebug::Print(_L("CMMFSwCodecPlayDataPath::Start-Normal"));
sl@0
   165
#endif
sl@0
   166
		// get sample rate and channels from RMdaDevSound
sl@0
   167
		RMdaDevSound::TCurrentSoundFormatBuf format;
sl@0
   168
		iSoundDevice.GetPlayFormat(format);
sl@0
   169
		iSampleRate = format().iRate;
sl@0
   170
		iChannels = format().iChannels;
sl@0
   171
		
sl@0
   172
		iNoMoreSourceData = EFalse;
sl@0
   173
		iNoMoreSoundDeviceData = EFalse;
sl@0
   174
		iSourceBuffer->SetLastBuffer(EFalse);
sl@0
   175
		iBytesPlayed = 0;
sl@0
   176
		iState = EPlaying;
sl@0
   177
		iSoundDeviceErrorReceiver->Start();
sl@0
   178
		TRAP(startError,FillSourceBufferL()); //get initial buffer of audio data
sl@0
   179
		if (startError == KErrNone)
sl@0
   180
			{
sl@0
   181
			// Start the player objects
sl@0
   182
			iAudioPlayer->Start();
sl@0
   183
			}
sl@0
   184
		else
sl@0
   185
			{//failed to start up correctly go back to stopped state
sl@0
   186
			iState = EStopped;
sl@0
   187
			iSoundDeviceErrorReceiver->Stop();
sl@0
   188
			}
sl@0
   189
   		}
sl@0
   190
	return startError;
sl@0
   191
	}
sl@0
   192
sl@0
   193
sl@0
   194
// *** Main Play Loop ***
sl@0
   195
sl@0
   196
void CMMFSwCodecPlayDataPath::FillSourceBufferL()
sl@0
   197
	{//asks observer to fill the source buffer          
sl@0
   198
    // Ask immediately for data from the observer
sl@0
   199
#ifdef __CYCLE_MMF_DATABUFFERS__
sl@0
   200
	// Create a new buffer to replicate INC021405 Play-EOF-Play on HwAccelerated solution Panics
sl@0
   201
	// If the creation fails, we carry on regardless as the original buffer will not have been 
sl@0
   202
	// destroyed. Must do this as alloc fail tests will not run.
sl@0
   203
	if(iSourceBuffer)
sl@0
   204
		{
sl@0
   205
		iSourceBuffer = CycleAudioBuffer(iSourceBuffer);
sl@0
   206
		}
sl@0
   207
#endif // __CYCLE_MMF_DATABUFFERS__	
sl@0
   208
	User::LeaveIfError(iHwDeviceObserver->FillThisHwBuffer(*iSourceBuffer));
sl@0
   209
	
sl@0
   210
	}
sl@0
   211
sl@0
   212
sl@0
   213
void CMMFSwCodecPlayDataPath::BufferFilledL(CMMFDataBuffer& aBuffer)
sl@0
   214
	{//call back from observer to indicate buffer has been filled
sl@0
   215
	if (iState == EStopped)
sl@0
   216
		User::Leave(KErrNotReady);//ok if paused?
sl@0
   217
sl@0
   218
	iSourceBuffer = &aBuffer;
sl@0
   219
	iSourceBuffer->SetStatus(EFull);
sl@0
   220
#ifdef _SCW_DEBUG
sl@0
   221
	RDebug::Print(_L("CMMFSwCodecPlayDataPath::BufferFilledL"));
sl@0
   222
#endif
sl@0
   223
sl@0
   224
	//need to check that the buffer size is not 0 - if so assume we've reached the end of the data
sl@0
   225
	if (!iSourceBuffer->BufferSize())
sl@0
   226
		{//no buffer  - could be end of source or could be that the source has no data??
sl@0
   227
		iNoMoreSourceData = ETrue;
sl@0
   228
#ifdef _SCW_DEBUG
sl@0
   229
		RDebug::Print(_L("CMMFSwCodecPlayDataPath::BufferFilledL-NoMoreSourceData"));
sl@0
   230
#endif
sl@0
   231
		}
sl@0
   232
	//even if the buffer size is 0 we still 
sl@0
   233
	//need to perform the following to get the sound device callback
sl@0
   234
	FillSoundDeviceBufferL(); //get buffer in pcm16 format for sound device	
sl@0
   235
sl@0
   236
    /* iVbrFlag is added to datapath to avail the alternative dataflow wherein datapath makes sure that 
sl@0
   237
    destinationbuffer is filled to its maximum length before sending it to the sound driver. 
sl@0
   238
    Sending the buffer directly to the device causes underflow incase of Vorbis codecs.*/
sl@0
   239
    if (iVbrFlag)
sl@0
   240
    	{
sl@0
   241
    	/*There are two cases we need to deal here
sl@0
   242
      	1. When the output of the codec is 0 for header data.
sl@0
   243
           in that case, repeat till actual decoding of ogg packets and pages.
sl@0
   244
        2. When destination buffer is not filled even to its half length, get next source buffer
sl@0
   245
           and decode it. This is to avoid underflows when ever we receive little pcm for a 
sl@0
   246
           a given source buffer.
sl@0
   247
           */
sl@0
   248
	    if (iSoundDeviceBuffer->Data().Length() < iSoundDeviceBuffer->Data().MaxLength()/2 && !(iSoundDeviceBuffer->LastBuffer()))
sl@0
   249
			{
sl@0
   250
	    	iSourceBuffer->SetStatus(EAvailable); //source buffer is now available
sl@0
   251
	    	iSoundDeviceBuffer->SetPosition(iSoundDeviceBuffer->Data().Length());//this indicates the available space in the buffer to the codec
sl@0
   252
	    	FillSourceBufferL();	
sl@0
   253
	    	return;
sl@0
   254
	    	}
sl@0
   255
		else //data is sufficient to avoid underflows
sl@0
   256
			{
sl@0
   257
			iSoundDeviceBuffer->SetPosition(0);
sl@0
   258
			if(iSoundDeviceBuffer->Data().Length()==0 && iSoundDeviceBuffer->LastBuffer())
sl@0
   259
				{
sl@0
   260
				iNoMoreSoundDeviceData = ETrue;
sl@0
   261
				}
sl@0
   262
			}	
sl@0
   263
    	}
sl@0
   264
sl@0
   265
    // attenuate the amplitude of the samples if volume ramping has been changed
sl@0
   266
	// and is non-zero
sl@0
   267
	if (iCustomInterface)
sl@0
   268
		{
sl@0
   269
		TTimeIntervalMicroSeconds volumeRamp = iCustomInterface->VolumeRamp();
sl@0
   270
		if (volumeRamp != iVolumeRamp)
sl@0
   271
			{
sl@0
   272
			iVolumeRamp = volumeRamp;
sl@0
   273
			if (iVolumeRamp.Int64() != 0)
sl@0
   274
				{
sl@0
   275
				iUtility->ConfigAudioRamper(
sl@0
   276
					iVolumeRamp.Int64(), 
sl@0
   277
					iSampleRate, 
sl@0
   278
					iChannels);
sl@0
   279
				iRampAudioSample = ETrue;
sl@0
   280
				}
sl@0
   281
			else
sl@0
   282
				{
sl@0
   283
				iRampAudioSample = EFalse;
sl@0
   284
				}
sl@0
   285
sl@0
   286
			}
sl@0
   287
			if (iRampAudioSample)
sl@0
   288
				iRampAudioSample = iUtility->RampAudio(iSoundDeviceBuffer);
sl@0
   289
		}
sl@0
   290
sl@0
   291
	iAudioPlayer->PlayData(*iSoundDeviceBuffer); //play data to sound drivers
sl@0
   292
sl@0
   293
	if (iSourceBuffer->LastBuffer())//check last buffer flag
sl@0
   294
		{
sl@0
   295
		iNoMoreSourceData = ETrue;
sl@0
   296
#ifdef _SCW_DEBUG
sl@0
   297
		RDebug::Print(_L("CMMFSwCodecPlayDataPath::BufferFilledL-LBNoMoreSourceData"));
sl@0
   298
#endif
sl@0
   299
		}
sl@0
   300
	}
sl@0
   301
sl@0
   302
sl@0
   303
void CMMFSwCodecPlayDataPath::FillSoundDeviceBufferL()
sl@0
   304
	{//use CMMFSwCodec to fill the sound device buffer
sl@0
   305
	
sl@0
   306
	CMMFSwCodec::TCodecProcessResult codecProcessResult;
sl@0
   307
sl@0
   308
	if (iCodec->IsNullCodec())
sl@0
   309
		{//no codec so data can be sent direct to sink
sl@0
   310
		iSoundDeviceBuffer = iSourceBuffer;
sl@0
   311
		iSoundDeviceBuffer->SetStatus(EFull);	//sink buffer is full
sl@0
   312
		}	
sl@0
   313
	else 
sl@0
   314
		{	
sl@0
   315
		//pass buffer to codec for processing
sl@0
   316
		codecProcessResult = iCodec->ProcessL(*iSourceBuffer, *iSoundDeviceBuffer);
sl@0
   317
		
sl@0
   318
		if (iSourceBuffer->LastBuffer()) //if source is last buffer so is sound dev
sl@0
   319
			iSoundDeviceBuffer->SetLastBuffer(ETrue);
sl@0
   320
		if ((!iSoundDeviceBuffer->BufferSize())&&(codecProcessResult.iDstBytesAdded))
sl@0
   321
			{//the codec has added data but not set the buffer length
sl@0
   322
			iSoundDeviceBuffer->Data().SetLength(codecProcessResult.iDstBytesAdded);
sl@0
   323
			}
sl@0
   324
		//only supports EProcessComplete
sl@0
   325
		switch (codecProcessResult.iCodecProcessStatus)
sl@0
   326
			{
sl@0
   327
		case CMMFSwCodec::TCodecProcessResult::EProcessComplete:
sl@0
   328
		//finished procesing source data - all data in sink buffer
sl@0
   329
			{
sl@0
   330
			iSoundDeviceBuffer->SetStatus(EFull);	//sink buffer is full	
sl@0
   331
			}
sl@0
   332
		break;
sl@0
   333
#ifdef SYMBIAN_VARIABLE_BITRATE_CODEC
sl@0
   334
		case CMMFSwCodec::TCodecProcessResult::EProcessIncomplete:
sl@0
   335
		//finished procesing source data - all data in sink buffer
sl@0
   336
			{
sl@0
   337
			iSoundDeviceBuffer->SetStatus(EFull);	//sink buffer is full	
sl@0
   338
			}
sl@0
   339
		break;
sl@0
   340
#endif
sl@0
   341
		case CMMFSwCodec::TCodecProcessResult::EDstNotFilled:
sl@0
   342
		//could be the last buffer in which case dst might not get filled
sl@0
   343
			{
sl@0
   344
			iSoundDeviceBuffer->SetStatus(EFull);	//sink buffer is full	
sl@0
   345
			}
sl@0
   346
		break;
sl@0
   347
		case CMMFSwCodec::TCodecProcessResult::EEndOfData:
sl@0
   348
			//no more data - send what we've got to the sink
sl@0
   349
			//note we can't always rely on this  - in many cases the codec will not know when
sl@0
   350
			//it has reached the end of data.
sl@0
   351
			{
sl@0
   352
			iSoundDeviceBuffer->SetStatus(EFull);//sink buffer may not really be 'full' but its as full as it going to get
sl@0
   353
			iNoMoreSourceData = ETrue;
sl@0
   354
			//doesn't matter if sink buffer is not full
sl@0
   355
			}
sl@0
   356
		break;
sl@0
   357
		default:
sl@0
   358
			Panic(EMMFSwCodecWrapperBadCodec); //should never get here - bad codec
sl@0
   359
			}
sl@0
   360
		}
sl@0
   361
	}
sl@0
   362
sl@0
   363
sl@0
   364
void CMMFSwCodecPlayDataPath::BufferEmptiedL(const CMMFDataBuffer& aBuffer)
sl@0
   365
	{//call back from CDataPathPlayer when the sound device buffer has been emptied
sl@0
   366
	if (&aBuffer != iSoundDeviceBuffer) 
sl@0
   367
		Panic(EMMFSwCodecWrapperBadBuffer);
sl@0
   368
	if(iVbrFlag && (iSourceBuffer->Status() == EUnAvailable || iNoMoreSourceData))
sl@0
   369
		{//No more source data. Play rest of the decoded data.Inform codec not to consider the source buffer
sl@0
   370
		if(iSourceBuffer->Status()!=EUnAvailable)
sl@0
   371
			{
sl@0
   372
			iSourceBuffer->SetStatus(EUnAvailable);
sl@0
   373
			}
sl@0
   374
		FillSoundDeviceBufferL();
sl@0
   375
		if(iSoundDeviceBuffer->BufferSize() > 0)
sl@0
   376
			{
sl@0
   377
			// attenuate the amplitude of the samples if volume ramping has been changed
sl@0
   378
			// and is non-zero
sl@0
   379
			if (iCustomInterface)
sl@0
   380
				{
sl@0
   381
				TTimeIntervalMicroSeconds volumeRamp = iCustomInterface->VolumeRamp();
sl@0
   382
				if (volumeRamp != iVolumeRamp)
sl@0
   383
					{
sl@0
   384
					iVolumeRamp = volumeRamp;
sl@0
   385
					if (iVolumeRamp.Int64() != 0)
sl@0
   386
						{
sl@0
   387
						iUtility->ConfigAudioRamper(iVolumeRamp.Int64(), iSampleRate, iChannels);
sl@0
   388
						iRampAudioSample = ETrue;
sl@0
   389
						}
sl@0
   390
					else
sl@0
   391
						{
sl@0
   392
						iRampAudioSample = EFalse;
sl@0
   393
						}
sl@0
   394
sl@0
   395
					}
sl@0
   396
				if (iRampAudioSample)
sl@0
   397
					{
sl@0
   398
					iRampAudioSample = iUtility->RampAudio(iSoundDeviceBuffer);
sl@0
   399
					}
sl@0
   400
					
sl@0
   401
				}
sl@0
   402
			iAudioPlayer->PlayData(*iSoundDeviceBuffer); //play data to sound drivers
sl@0
   403
			return;
sl@0
   404
			}
sl@0
   405
		else
sl@0
   406
			{
sl@0
   407
			if(iNoMoreSourceData)
sl@0
   408
				{
sl@0
   409
				iNoMoreSoundDeviceData = ETrue;
sl@0
   410
				}
sl@0
   411
			iSourceBuffer->SetStatus(EAvailable);
sl@0
   412
			}
sl@0
   413
		}
sl@0
   414
	if (!iNoMoreSourceData) 
sl@0
   415
		FillSourceBufferL();
sl@0
   416
	}
sl@0
   417
sl@0
   418
//*** End of Main Play Loop ***
sl@0
   419
sl@0
   420
sl@0
   421
void CMMFSwCodecPlayDataPath::Stop()
sl@0
   422
	{
sl@0
   423
	iAudioPlayer->Cancel();
sl@0
   424
	iSoundDeviceErrorReceiver->Cancel();
sl@0
   425
    iSoundDevice.Close();
sl@0
   426
sl@0
   427
#ifdef __CYCLE_MMF_DATABUFFERS__
sl@0
   428
	// Create a new buffer to replicate INC021405 Play-EOF-Play on HwAccelerated solution Panics
sl@0
   429
	// If the creation fails, we carry on regardless as the original buffer will not have been 
sl@0
   430
	// destroyed. Must do this as alloc fail tests will not run.
sl@0
   431
	if(iSourceBuffer)
sl@0
   432
		{
sl@0
   433
		iSourceBuffer = CycleAudioBuffer(iSourceBuffer);
sl@0
   434
		}
sl@0
   435
#endif // __CYCLE_MMF_DATABUFFERS__	
sl@0
   436
sl@0
   437
	iState = EStopped;
sl@0
   438
	}
sl@0
   439
sl@0
   440
sl@0
   441
void CMMFSwCodecPlayDataPath::Pause()
sl@0
   442
	{
sl@0
   443
	//since a pause can happen anyway in the datatransfer -need to set to a known 
sl@0
   444
	//state so that when play is resumed the behaviour is predictable
sl@0
   445
	if (iSoundDevice.Handle())
sl@0
   446
		{
sl@0
   447
		iSoundDevice.PausePlayBuffer(); //needs new LDD
sl@0
   448
		iState = EPaused;
sl@0
   449
#ifdef _SCW_DEBUG
sl@0
   450
		RDebug::Print(_L("Pause"));
sl@0
   451
#endif
sl@0
   452
		}
sl@0
   453
	else
sl@0
   454
		{//an error must have occured 
sl@0
   455
		iState = EStopped;
sl@0
   456
		}
sl@0
   457
	}
sl@0
   458
	
sl@0
   459
	
sl@0
   460
TInt CMMFSwCodecPlayDataPath::EmptyBuffers()
sl@0
   461
	{
sl@0
   462
	TInt error = KErrNone;
sl@0
   463
	if (iSoundDevice.Handle() == 0)
sl@0
   464
		{
sl@0
   465
		error = KErrNotReady;
sl@0
   466
		}
sl@0
   467
	else
sl@0
   468
		{ 
sl@0
   469
		iAudioPlayer->Cancel();
sl@0
   470
		iSoundDevice.CancelPlayData();
sl@0
   471
		iSoundDeviceErrorReceiver->Stop();
sl@0
   472
		iState = EStopped;
sl@0
   473
		}
sl@0
   474
	return error;
sl@0
   475
	}	
sl@0
   476
sl@0
   477
sl@0
   478
RMdaDevSound& CMMFSwCodecPlayDataPath::Device()
sl@0
   479
	{
sl@0
   480
	return iSoundDevice;
sl@0
   481
	}
sl@0
   482
sl@0
   483
sl@0
   484
void CMMFSwCodecPlayDataPath::SoundDeviceException(TInt aError)
sl@0
   485
	{
sl@0
   486
	if(iIgnoreUnderflow)
sl@0
   487
		{
sl@0
   488
		if(!iVbrFlag && aError==KErrUnderflow && !iNoMoreSourceData)
sl@0
   489
			{
sl@0
   490
			//ignore underflow
sl@0
   491
			return;
sl@0
   492
			}
sl@0
   493
		//for VBR codec data,no more source does not mean that no more sounddevice data
sl@0
   494
		//so ignore underflows till the last buffer is played from the codec
sl@0
   495
		else if(iVbrFlag && aError==KErrUnderflow && !iNoMoreSoundDeviceData)
sl@0
   496
			{
sl@0
   497
			//ignore underflow
sl@0
   498
			return;
sl@0
   499
			}	
sl@0
   500
		}
sl@0
   501
	
sl@0
   502
	//this sends a request to the hw device observer usually Devsound
sl@0
   503
	//to update the bytes played
sl@0
   504
	//it is done here so that the sound driver can be closed prior to
sl@0
   505
	//updating the plicy and sending the error back
sl@0
   506
	TUid uidUpdateBytesPlayed;
sl@0
   507
	uidUpdateBytesPlayed.iUid = KMmfHwDeviceObserverUpdateBytesPlayed;
sl@0
   508
	TPtrC8 dummy(0,0);
sl@0
   509
	iHwDeviceObserver->MsgFromHwDevice(uidUpdateBytesPlayed,dummy);
sl@0
   510
sl@0
   511
	//this closes RMdaDevSound.
sl@0
   512
	Stop(); 
sl@0
   513
sl@0
   514
	//inform devsound so it can update policy
sl@0
   515
	iHwDeviceObserver->Stopped(); 
sl@0
   516
sl@0
   517
	// Inform the observer of the exception condition
sl@0
   518
	// We inform the hw device observer after the policy has been
sl@0
   519
	// updated incase the observer relied on the error to assume
sl@0
   520
	// the policy has been updated
sl@0
   521
	iHwDeviceObserver->Error(aError);
sl@0
   522
	}
sl@0
   523
sl@0
   524
sl@0
   525
void CMMFSwCodecPlayDataPath::SetPlayCustomInterface(MPlayCustomInterface& aCustomInterface)
sl@0
   526
	{
sl@0
   527
	iCustomInterface = &aCustomInterface;
sl@0
   528
	}
sl@0
   529
	
sl@0
   530
/**
sl@0
   531
Retrieves a custom interface to the device.
sl@0
   532
The reference CMMFSwCodecWrapper supports two  custom interfaces,
sl@0
   533
MEmptyBuffersCustomInterface and MSetVbrFlagCustomInterface
sl@0
   534
sl@0
   535
@param	aInterface
sl@0
   536
		Interface UID, defined with the custom interface.
sl@0
   537
		aInterface = KMmfUidEmptyBuffersCustomInterface for MEmptyBuffersCustomInterface,
sl@0
   538
					 KSetVbrFlagCustomInterfaceTypeUid for MSetVbrFlagCustomInterface
sl@0
   539
		
sl@0
   540
@return A pointer to the interface implementation, or NULL if the device can not
sl@0
   541
		implement the interface requested. The return value must be cast to the
sl@0
   542
		correct type by the user.
sl@0
   543
*/
sl@0
   544
TAny* CMMFSwCodecPlayDataPath::CustomInterface(TUid aInterface)
sl@0
   545
	{
sl@0
   546
	TAny* ret = NULL;
sl@0
   547
	if (aInterface.iUid == KMmfUidEmptyBuffersCustomInterface)
sl@0
   548
		{
sl@0
   549
		MEmptyBuffersCustomInterface* result = static_cast<MEmptyBuffersCustomInterface*> (this);
sl@0
   550
		ret = static_cast<TAny*>(result);
sl@0
   551
		}
sl@0
   552
	else if(aInterface.iUid == KSetVbrFlagCustomInterfaceTypeUid)
sl@0
   553
		{
sl@0
   554
		SetVbrFlag();
sl@0
   555
		}
sl@0
   556
	if (aInterface == KTimePlayedCustomInterfaceTypeUid)
sl@0
   557
		{
sl@0
   558
		MTimePlayedCustomInterface* result = static_cast<MTimePlayedCustomInterface*> (this);
sl@0
   559
		ret = static_cast<TAny*>(result);
sl@0
   560
		}
sl@0
   561
	if (aInterface == KIgnoreUnderflowCustomInterfaceTypeUid)
sl@0
   562
		{
sl@0
   563
		MIgnoreUnderflowEventsCustomInterface* result = static_cast<MIgnoreUnderflowEventsCustomInterface*> (this);
sl@0
   564
		ret = static_cast<TAny*>(result);
sl@0
   565
		}
sl@0
   566
	return ret;
sl@0
   567
	}
sl@0
   568
sl@0
   569
/**
sl@0
   570
Used to set iVbrFlag on the datapath.
sl@0
   571
sl@0
   572
This method is used to set the iVbrFlag in datapath. This flag is added to datapath to avail the 
sl@0
   573
alternative dataflow wherein datapath makes sure that destinationbuffer is filled to its maximum length
sl@0
   574
before sending it to the sound driver. Sending the buffer directly to the device causes underflow incase of VBR codecs.
sl@0
   575
*/		
sl@0
   576
void CMMFSwCodecPlayDataPath::SetVbrFlag()
sl@0
   577
	{
sl@0
   578
	iVbrFlag = ETrue;
sl@0
   579
	}
sl@0
   580
sl@0
   581
TInt CMMFSwCodecPlayDataPath::GetTimePlayed(TTimeIntervalMicroSeconds& aTime)
sl@0
   582
	{
sl@0
   583
	if(iSoundDevice.Handle())
sl@0
   584
		{
sl@0
   585
		TInt bytes = iSoundDevice.BytesPlayed();
sl@0
   586
		//Work around for overflow of bytes played from driver.
sl@0
   587
		//This code will be removed when Base provides the TimePlayed() API which returns the play time
sl@0
   588
		//Assuming GetTimePlayed() gets called in an interval not more than 3 secs, reset driver's bytes played when it is near KMaxInt
sl@0
   589
		if(bytes > (KMaxTInt - 3*KMaxBytesInSec))
sl@0
   590
			{
sl@0
   591
			iBytesPlayed = iBytesPlayed+bytes;
sl@0
   592
			iSoundDevice.ResetBytesPlayed();
sl@0
   593
			bytes = 0;
sl@0
   594
			}
sl@0
   595
		TInt64 samplesPlayed = (iBytesPlayed+bytes)/(KBytesPerSample*iChannels);
sl@0
   596
		aTime = (samplesPlayed*1000000)/iSampleRate;
sl@0
   597
		}
sl@0
   598
	else
sl@0
   599
		{
sl@0
   600
		aTime = 0;
sl@0
   601
		}
sl@0
   602
		
sl@0
   603
	return KErrNone;
sl@0
   604
	}
sl@0
   605
sl@0
   606
void CMMFSwCodecPlayDataPath::IgnoreUnderflowEvents()
sl@0
   607
	{
sl@0
   608
	iIgnoreUnderflow = ETrue;
sl@0
   609
	}
sl@0
   610
/************************************************************************
sl@0
   611
 *				CDataPathPlayer											*
sl@0
   612
 ************************************************************************/
sl@0
   613
sl@0
   614
CDataPathPlayer::CDataPathPlayer(CMMFSwCodecPlayDataPath& aParent, TInt aPriority)
sl@0
   615
: CActive(aPriority), iParent(aParent)
sl@0
   616
	{
sl@0
   617
	CActiveScheduler::Add(this);
sl@0
   618
	}
sl@0
   619
sl@0
   620
sl@0
   621
CDataPathPlayer::~CDataPathPlayer()
sl@0
   622
	{
sl@0
   623
	Cancel();
sl@0
   624
	}
sl@0
   625
sl@0
   626
sl@0
   627
void CDataPathPlayer::Start()
sl@0
   628
	{
sl@0
   629
	// No implementation
sl@0
   630
	}
sl@0
   631
sl@0
   632
sl@0
   633
void CDataPathPlayer::ResumePlaying()
sl@0
   634
	{
sl@0
   635
	if (iParent.Device().Handle())
sl@0
   636
		{
sl@0
   637
		//should be ok to call this even if we are active
sl@0
   638
		iParent.Device().ResumePlaying(); 
sl@0
   639
		iResumePlaying = ETrue;
sl@0
   640
		}
sl@0
   641
#ifdef _SCW_DEBUG
sl@0
   642
	RDebug::Print(_L("Playing Resumed"));
sl@0
   643
#endif
sl@0
   644
	}
sl@0
   645
sl@0
   646
sl@0
   647
void CDataPathPlayer::PlayData(const CMMFDataBuffer& aData)
sl@0
   648
	{
sl@0
   649
	iDataFromSource = &aData;
sl@0
   650
	if (!IsActive())
sl@0
   651
		{
sl@0
   652
#ifdef _SCW_DEBUG
sl@0
   653
		RDebug::Print(_L("CDataPathPlayer::PlayData"));
sl@0
   654
#endif
sl@0
   655
		iParent.Device().PlayData(iStatus,(STATIC_CAST(const CMMFDataBuffer*, iDataFromSource))->Data());
sl@0
   656
		SetActive();
sl@0
   657
		}
sl@0
   658
	}
sl@0
   659
sl@0
   660
sl@0
   661
void CDataPathPlayer::Stop()
sl@0
   662
	{
sl@0
   663
	if (!IsActive())	
sl@0
   664
		iParent.Device().FlushPlayBuffer(); // Otherwise won't be flushed
sl@0
   665
	Cancel();
sl@0
   666
	iParent.SoundDeviceException(KErrCancel);
sl@0
   667
	}
sl@0
   668
sl@0
   669
sl@0
   670
void CDataPathPlayer::RunL()
sl@0
   671
	{
sl@0
   672
#ifdef _SCW_DEBUG
sl@0
   673
	RDebug::Print(_L("CDataPathPlayer::RunL error[%d]"), iStatus.Int());
sl@0
   674
#endif
sl@0
   675
	if (iStatus.Int()!=KErrNone)
sl@0
   676
		{ 	
sl@0
   677
		iParent.SoundDeviceException(iStatus.Int());
sl@0
   678
		}
sl@0
   679
	else
sl@0
   680
		{
sl@0
   681
		iParent.BufferEmptiedL(static_cast<const CMMFDataBuffer&>(*iDataFromSource));
sl@0
   682
		iResumePlaying = EFalse;
sl@0
   683
		}
sl@0
   684
	}
sl@0
   685
sl@0
   686
sl@0
   687
TInt CDataPathPlayer::RunError(TInt aError)
sl@0
   688
	{
sl@0
   689
	Error(aError);
sl@0
   690
	return KErrNone;
sl@0
   691
	}
sl@0
   692
sl@0
   693
sl@0
   694
void CDataPathPlayer::DoCancel()
sl@0
   695
	{
sl@0
   696
	if (iParent.Device().Handle())
sl@0
   697
		{
sl@0
   698
		iParent.Device().CancelPlayData();
sl@0
   699
		iParent.Device().FlushPlayBuffer();
sl@0
   700
		}
sl@0
   701
	}
sl@0
   702
sl@0
   703
sl@0
   704
void CDataPathPlayer::Error(TInt aError)
sl@0
   705
	{ 
sl@0
   706
	iParent.SoundDeviceException(aError);
sl@0
   707
	}
sl@0
   708
sl@0
   709
sl@0
   710
/************************************************************************
sl@0
   711
 *				CSoundDevPlayErrorReceiver											*
sl@0
   712
 ************************************************************************/
sl@0
   713
sl@0
   714
CSoundDevPlayErrorReceiver::CSoundDevPlayErrorReceiver(CMMFSwCodecPlayDataPath& aParent, TInt aPriority)
sl@0
   715
: CActive(aPriority), iParent(aParent)
sl@0
   716
	{
sl@0
   717
	CActiveScheduler::Add(this);
sl@0
   718
	}
sl@0
   719
sl@0
   720
CSoundDevPlayErrorReceiver::~CSoundDevPlayErrorReceiver()
sl@0
   721
	{
sl@0
   722
	Cancel();
sl@0
   723
	}
sl@0
   724
sl@0
   725
void CSoundDevPlayErrorReceiver::Start()
sl@0
   726
	{
sl@0
   727
	iParent.Device().NotifyPlayError(iStatus);
sl@0
   728
	SetActive();
sl@0
   729
	}
sl@0
   730
sl@0
   731
void CSoundDevPlayErrorReceiver::Stop()
sl@0
   732
	{
sl@0
   733
	Cancel();
sl@0
   734
	}
sl@0
   735
sl@0
   736
void CSoundDevPlayErrorReceiver::RunL()
sl@0
   737
	{
sl@0
   738
	TInt reason = iStatus.Int();
sl@0
   739
	Start();
sl@0
   740
	// An error has been returned
sl@0
   741
#ifdef _SCW_DEBUG
sl@0
   742
	RDebug::Print(_L("CSoundDevPlayErrorReceiver::RunL[%d]"), reason);
sl@0
   743
#endif
sl@0
   744
	iParent.SoundDeviceException(reason);
sl@0
   745
	}
sl@0
   746
sl@0
   747
void CSoundDevPlayErrorReceiver::DoCancel()
sl@0
   748
	{
sl@0
   749
	iParent.Device().CancelNotifyPlayError();
sl@0
   750
	}
sl@0
   751
sl@0
   752