os/mm/devsound/devsoundrefplugin/src/swcodecwrapper/mmfSwCodecRecordDataPath.cpp
author sl@SLION-WIN7.fritz.box
Fri, 15 Jun 2012 03:10:57 +0200
changeset 0 bde4ae8d615e
permissions -rw-r--r--
First public contribution.
     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".
     7 //
     8 // Initial Contributors:
     9 // Nokia Corporation - initial contribution.
    10 //
    11 // Contributors:
    12 //
    13 // Description:
    14 // source\server\mmfswcodecrecorddatapath.cpp
    15 //
    16 //
    17 
    18 #include "mmfSwCodecRecordDataPath.h"
    19 #include <mmf/server/mmfswcodecwrapper.h>
    20 #include <mmf/common/mmfpaniccodes.h>
    21 
    22 #ifdef SYMBIAN_SCW_DEBUG
    23 
    24 const TText* const KStateNames[] = // must agree with TRecordState
    25             {
    26             _S("ERecordStateCreated"),
    27             _S("ERecordStateFailed"),
    28             _S("ERecordStateRecording"),
    29             _S("ERecordStateSendingBuffer"),
    30             _S("ERecordStateSendingPartialBuffer"),
    31             _S("ERecordStateEmptiedPartialBuffer"),
    32             _S("ERecordStateRecordingPaused"),
    33             _S("ERecordStateSendingBufferPaused"),
    34             _S("ERecordStateSendingPartialBufferPaused"),
    35             _S("ERecordStateEmptiedPartialBufferPaused"),
    36             };
    37 
    38 static const TText* StateName(TInt aState)
    39     {
    40     return KStateNames[aState];
    41     }
    42 
    43 #endif // SYMBIAN_SCW_DEBUG
    44 
    45 // Table of next state used when resuming or pausing.
    46 const CMMFSwCodecRecordDataPath::TRecordState CMMFSwCodecRecordDataPath::KResumePauseTable[] =
    47     {
    48     ERecordStateCreated,                    //ERecordStateCreated                       // note order here is important - see State(), RecordOrPause() etc
    49     ERecordStateFailed,                     //ERecordStateFailed
    50     ERecordStateRecordingPaused,            //ERecordStateRecording
    51     ERecordStateSendingBufferPaused,        //ERecordStateSendingBuffer
    52     ERecordStateSendingPartialBufferPaused, //ERecordStateSendingPartialBuffer
    53     ERecordStateEmptiedPartialBufferPaused, //ERecordStateEmptiedPartialBuffer
    54     ERecordStateRecording,                  //ERecordStateRecordingPaused
    55     ERecordStateSendingBuffer,              //ERecordStateSendingBufferPaused
    56     ERecordStateSendingPartialBuffer,       //ERecordStateSendingPartialBufferPaused
    57     ERecordStateEmptiedPartialBuffer,       //ERecordStateEmptiedPartialBufferPaused
    58     };
    59 
    60 
    61 CMMFSwCodecRecordDataPath* CMMFSwCodecRecordDataPath::NewL()
    62 	{
    63 	CMMFSwCodecRecordDataPath* self = new(ELeave) CMMFSwCodecRecordDataPath;
    64 	CleanupStack::PushL(self);
    65 	self->ConstructL();
    66 	CleanupStack::Pop();
    67 	return self;
    68 	}
    69 
    70 
    71 void CMMFSwCodecRecordDataPath::ConstructL()
    72 	{
    73 	iAudioInput = MAudioInput::CreateL(*this);
    74 	TCallBack callback(Callback, this);
    75 	iAsyncCallback = new (ELeave) CAsyncCallBack(callback, CActive::EPriorityStandard);
    76 	}
    77 
    78 CMMFSwCodecRecordDataPath::CMMFSwCodecRecordDataPath():
    79 		iShadowData(NULL, 0, 0)
    80 	{
    81 	ASSERT(iState==ERecordStateCreated); // assume this value is 0, so no need to assign
    82 	}
    83 
    84 CMMFSwCodecRecordDataPath::~CMMFSwCodecRecordDataPath()
    85 	{
    86 	if (iAudioInput)
    87 		{
    88 		iAudioInput->Release();
    89 		}
    90 
    91 	delete iCodecBuffer;
    92 	delete iInputBuffer;
    93 	delete iAsyncCallback;
    94 	}
    95 
    96 
    97 TInt CMMFSwCodecRecordDataPath::SetObserver(MMMFHwDeviceObserver& aObserver)
    98 	{
    99 	TInt error;
   100 	if (iHwDeviceObserver)
   101 		{
   102 		error =  KErrAlreadyExists;
   103 		}
   104 	else
   105 		{
   106 		iHwDeviceObserver = &aObserver;
   107 		error  = KErrNone;
   108 		}
   109 	return error;
   110 	}
   111 
   112 
   113 TInt CMMFSwCodecRecordDataPath::AddCodec(CMMFSwCodec& aCodec)
   114 	{
   115 	TInt err = KErrNone;
   116 
   117 	if (iCodec)
   118 		{
   119 		err = KErrNotSupported; //doesn't support multiple codecs
   120 		}
   121 
   122 	if (!err)
   123 		{
   124 		iCodec = &aCodec;
   125 
   126 		iSinkBufferSize = iCodec->SinkBufferSize();
   127 		iAudioInputBufferSize = iCodec->SourceBufferSize(); // the buffer size we want from the input device
   128 
   129 		if (!iSinkBufferSize || !iAudioInputBufferSize)
   130 			{
   131 			err = KErrArgument; //codec plugin has not specified buffer size
   132 			}
   133 		}
   134 
   135 	if (!err)
   136 		{
   137 		// Allocate data buffer
   138 		if (iCodec->IsNullCodec())
   139 			{//don't need a separate sink buffer if null codec
   140 			iSinkBuffer = NULL; //sink buffer is the sound device buffer
   141 			iAudioInputBufferSize = iSinkBufferSize; // the audio input buffer becomes the sink buffer
   142 			}
   143 		else
   144 			{//need a separate sink buffer for the codec - this is the buffer passed to our client
   145 			ASSERT(!iCodecBuffer); // can't happen because can only call AddCodec once
   146 			TRAP(err,iCodecBuffer = CMMFDataBuffer::NewL(iSinkBufferSize));
   147 			iSinkBuffer = iCodecBuffer;
   148 			}
   149 		}
   150 	if (!err)
   151 		{
   152 		ASSERT(!iInputBuffer); // can't happen because can only call AddCodec once
   153 		TRAP(err,iInputBuffer = CMMFPtrBuffer::NewL());
   154 		}
   155 	if (!err)
   156 		{
   157 		// point iSinkBuffer at the right place
   158 		if (iCodec->IsNullCodec())
   159 			{
   160 			iSinkBuffer = iInputBuffer;
   161 			}
   162 		else
   163 			{
   164 			iSinkBuffer = iCodecBuffer;
   165 			}
   166 		}
   167 	return err;
   168 	}
   169 
   170 
   171 TInt CMMFSwCodecRecordDataPath::Start()
   172 	{
   173 #ifdef SYMBIAN_SCW_DEBUG
   174     RDebug::Print(_L("CMMFSwcodecRecordDataPath::Start() state=%s"), StateName(iState));
   175 #endif
   176 	TInt err = KErrNone;
   177 	if (!iCodec)
   178 		{//check that a codec has been added
   179 		err = KErrNotReady;
   180 		}
   181 	if (!err)
   182 		{
   183 		switch (iState)
   184 		    {
   185 		    case ERecordStateCreated:
   186 		        {
   187 	            TAudioInputParams params;
   188 	            params.iInitialGain = iGain;
   189 	            params.iSampleRate = iSampleRate;
   190 	            params.iNumChannels = iNumChannels;
   191 	            params.iNominalBufferSize = iAudioInputBufferSize;
   192 	            err = iAudioInput->Initialize(params);
   193                 if (!err)
   194                     {
   195                     err = iAudioInput->Start();
   196                     if (err)
   197                         {
   198                         iAudioInput->Close();
   199                         ASSERT(iState == ERecordStateCreated); // end up in same state
   200                         }
   201                     else
   202                         {
   203                         iState = ERecordStateRecording;
   204                         iInputHasFinished = EFalse;
   205                         iRecordedBytesCount = 0; //used for debug purposes
   206                         }
   207                     }
   208 		        }
   209 		        break;
   210 		    case ERecordStateRecordingPaused:
   211 		    case ERecordStateSendingPartialBufferPaused:
   212 		    case ERecordStateEmptiedPartialBufferPaused:
   213 		        {
   214 		        // effectively in paused state, resume and switch to equivalent state
   215 		        iAudioInput->Resume();
   216                 iInputHasFinished = EFalse;
   217 		        iState = KResumePauseTable[iState];
   218 		        }
   219 		        break;
   220             case ERecordStateSendingBufferPaused:
   221                 {
   222 				iAudioInput->Resume();
   223                 if (iInputHasFinished)
   224                     {
   225                     iState = ERecordStateRecording; // as we follow InputHasFinished, we don't wait for the buffer
   226 					iInputHasFinished = EFalse;
   227                     }
   228                 else
   229                     {
   230                     // effectively in paused state, resume and switch to equivalent state
   231                     iState = KResumePauseTable[iState];
   232                     }
   233                 }
   234                 break;
   235             case ERecordStateFailed:
   236 		    default:
   237 		        {
   238 		        // anything else assume already recording and ignore
   239 		        }
   240 		        break;
   241 		    }
   242 		}
   243 
   244 #ifdef SYMBIAN_SCW_DEBUG
   245     RDebug::Print(_L("End CMMFSwcodecRecordDataPath::Start(%d) state=%s"), err, StateName(iState));
   246 #endif
   247 	return err;
   248 	}
   249 
   250 
   251 void CMMFSwCodecRecordDataPath::InputBufferAvailable(const TDesC8& aBuffer)
   252 	{
   253 #ifdef SYMBIAN_SCW_DEBUG
   254     RDebug::Print(_L("CMMFSwcodecRecordDataPath::InputBufferAvailable(%d) state=%s"), aBuffer.Length(), StateName(iState));
   255 #endif
   256 	ASSERT(iState==ERecordStateRecording || iState==ERecordStateRecordingPaused);
   257 	iInputData = &aBuffer;
   258     TUint length = aBuffer.Length();
   259 	// Update bytes recorded
   260 	iRecordedBytesCount += length;
   261 
   262 	//buffer ok can send to sink
   263 	iInputOffset = 0;
   264 	TRAPD(err,ProcessBufferL(EFalse)); //convert to sink data type using codec
   265 	if (err != KErrNone)
   266 	    {
   267 	    iHwDeviceObserver->Error(err);
   268 	    }
   269 #ifdef SYMBIAN_SCW_DEBUG
   270     RDebug::Print(_L("End CMMFSwcodecRecordDataPath::InputBufferAvailable state=%s"), StateName(iState));
   271 #endif
   272 	}
   273 
   274 void CMMFSwCodecRecordDataPath::InputFinished()
   275 	{
   276 #ifdef SYMBIAN_SCW_DEBUG
   277     RDebug::Print(_L("CMMFSwcodecRecordDataPath::InputFinished state=%s"), StateName(iState));
   278 #endif
   279     ASSERT(iState==ERecordStateRecording || iState==ERecordStateRecordingPaused);
   280 	iInputOffset = 0;
   281 	iInputHasFinished = ETrue;
   282 	TRAPD(err,ProcessBufferL(ETrue)); // finish off any conversion	
   283 	if (err != KErrNone)
   284 	    {
   285 	    iHwDeviceObserver->Error(err);
   286 	    }
   287 #ifdef SYMBIAN_SCW_DEBUG
   288     RDebug::Print(_L("End CMMFSwcodecRecordDataPath::InputFinished state=%s"), StateName(iState));
   289 #endif
   290 	}
   291 
   292 void CMMFSwCodecRecordDataPath::InputError(TInt aError)
   293 	{
   294 #ifdef SYMBIAN_SCW_DEBUG
   295     RDebug::Print(_L("CMMFSwcodecRecordDataPath::InputBufferAvailable(%d) state=%s"), aError, StateName(iState));
   296 #endif
   297 	if (iState!=ERecordStateFailed)
   298 	    {
   299 	    iState = ERecordStateFailed;
   300 	    if (iHwDeviceObserver)
   301 	        {
   302 	        // Inform the observer of the exception condition
   303 	        // Assume it will subsequently call Stop(), and thus update policy
   304 	        iHwDeviceObserver->Error(aError);
   305 	        }
   306 	    }
   307 #ifdef SYMBIAN_SCW_DEBUG
   308     RDebug::Print(_L("CMMFSwcodecRecordDataPath::InputBufferAvailable() state=%s"), StateName(iState));
   309 #endif
   310 	}
   311 
   312 
   313 /*
   314  *  FillSinkBufferL
   315  *
   316  *	Function to take the data from an already full source buffer and by using
   317  *	a codec if necessary fills the sink buffer
   318  *  If aLastBuffer, treat as a last buffer with zero length
   319  */
   320 void CMMFSwCodecRecordDataPath::ProcessBufferL(TBool aLastBuffer)
   321 	{
   322     ASSERT(iState==ERecordStateRecording || iState==ERecordStateRecordingPaused ||
   323            iState==ERecordStateEmptiedPartialBuffer || iState==ERecordStateEmptiedPartialBufferPaused); // only valid states
   324 	if (iCodec->IsNullCodec())
   325 		{//no codec so sound device buffer can be used directly as sink buffer
   326 		ASSERT(iSinkBuffer==iInputBuffer); // just assume this
   327 		if (aLastBuffer)
   328 			{
   329 			iShadowData.Set(NULL, 0, 0);
   330 			iInputBuffer->SetPtr(iShadowData);
   331 			iInputBuffer->SetLastBuffer(ETrue);
   332 			}
   333 		else
   334 			{
   335 			iShadowData.Set(const_cast<TUint8*>(iInputData->Ptr()), iInputData->Length(), iInputData->Length());
   336 			iInputBuffer->SetPtr(iShadowData);
   337 			iInputBuffer->SetLastBuffer(EFalse);
   338 			}
   339 		iInputBuffer->SetStatus(EFull);	//sink buffer is "full"
   340 		TRecordState oldState = iState;
   341         switch (iState)
   342             {
   343             case ERecordStateRecording:
   344                 iState = ERecordStateSendingBuffer;
   345                 break;
   346             case ERecordStateRecordingPaused:
   347                 iState = ERecordStateSendingBufferPaused;
   348                 break;
   349             case ERecordStateEmptiedPartialBuffer:
   350             case ERecordStateEmptiedPartialBufferPaused:
   351                 ASSERT(EFalse); // Technically these can occur but not if IsNullCodec is true, Complete is effectively always true
   352                 break;
   353             }
   354 		TInt err = iHwDeviceObserver->EmptyThisHwBuffer(*iSinkBuffer); //pass onto sink
   355 		if (err)
   356             {
   357             iState = oldState; // rewind
   358             User::Leave(err);
   359             }
   360 		}
   361 	else
   362 		{
   363 		ASSERT(iSinkBuffer==iCodecBuffer); // sink and codec buffers are synonym, so just talk to iSinkBuffer
   364 		if (aLastBuffer)
   365 			{
   366 			iShadowData.Set(NULL, 0, 0);
   367 			iInputBuffer->SetPtr(iShadowData); // empty buffer
   368 			iInputBuffer->SetLastBuffer(ETrue);
   369 			}
   370 		else
   371 			{
   372 			TPtrC8 tempData = iInputData->Mid(iInputOffset);
   373 			iShadowData.Set(const_cast<TUint8*>(tempData.Ptr()), tempData.Length(), tempData.Length());
   374 			iInputBuffer->SetPtr(iShadowData);
   375 			iInputBuffer->SetLastBuffer(EFalse);
   376 			}
   377 		//pass buffer to codec for processing
   378 		CMMFSwCodec::TCodecProcessResult codecProcessResult = iCodec->ProcessL(*iInputBuffer, *iSinkBuffer);
   379 		if ((!iSinkBuffer->BufferSize())&&(codecProcessResult.iDstBytesAdded))
   380 			{//the codec has added data but not set the buffer length
   381 			iSinkBuffer->Data().SetLength(codecProcessResult.iDstBytesAdded);
   382 			}
   383 		//only supports EProcessComplete
   384 		TRecordState oldState = iState;
   385 		TInt err = KErrNone;
   386 		switch (codecProcessResult.iCodecProcessStatus)
   387 			{
   388 			case CMMFSwCodec::TCodecProcessResult::EProcessComplete: //finished procesing source data - all data in sink buffer
   389 			case CMMFSwCodec::TCodecProcessResult::EDstNotFilled: //finished procesing source data - sink buffer not full could be EOF
   390 			case CMMFSwCodec::TCodecProcessResult::EEndOfData: //no more data - send what we've got to the sink
   391 				{
   392 				iSinkBuffer->SetStatus(EFull);	// treat sink buffer as full
   393 				iState = IsPaused() ? ERecordStateSendingBufferPaused : ERecordStateSendingBuffer;
   394 				err = EmptyBufferL();
   395 				break;
   396 				}
   397 			case CMMFSwCodec::TCodecProcessResult::EProcessIncomplete:
   398 				{
   399 				// codec has not yet finished with input - send buffer and come back around
   400 				iSinkBuffer->SetStatus(EFull);	// treat sink buffer as full
   401 				iInputOffset = codecProcessResult.iSrcBytesProcessed;
   402                 iState = IsPaused() ? ERecordStateSendingPartialBufferPaused : ERecordStateSendingPartialBuffer;
   403                 err = EmptyBufferL();
   404 				break;
   405 				}
   406 			default:
   407 				Panic(EMMFSwCodecWrapperBadCodec); //should never get here - bad codec
   408 			}
   409 		if (err)
   410 		    {
   411 		    iState = oldState; // rewind prior to handle
   412 		    User::Leave(err);
   413 		    }
   414 		}
   415 	}
   416 
   417 TInt CMMFSwCodecRecordDataPath::EmptyBufferL()
   418 	{
   419 	// This code supports an assumption made by the vorbis encoder, which assumes it can safely generate empty buffers.
   420 	// VbrFlag here implies the vorbis encoder.
   421 	// TODO: Replace this with a generic solution - e.g. on EDstNotFilled we request a buffer from AudioInput instead
   422 	// 		 of calling the client back.
   423 	if(iVbrFlag)
   424 		{
   425 		if(!iSinkBuffer->Data().Length() && !iInputBuffer->LastBuffer())
   426 			{
   427 			BufferEmptiedL(STATIC_CAST(CMMFDataBuffer&, *iSinkBuffer));
   428 			return KErrNone;
   429 			}
   430 		}
   431 	TInt err = iHwDeviceObserver->EmptyThisHwBuffer(*iSinkBuffer); //pass onto sink
   432 	return err;
   433 	}
   434 
   435 void CMMFSwCodecRecordDataPath::BufferEmptiedL(const CMMFDataBuffer& aBuffer)
   436 	{
   437 #ifdef SYMBIAN_SCW_DEBUG
   438     RDebug::Print(_L("CMMFSwcodecRecordDataPath::BufferEmptiedL() state=%s"), StateName(iState));
   439 #endif
   440 	if (&aBuffer != iSinkBuffer)
   441 		{
   442 		// we are only single buffering at the moment...
   443 		Panic(EMMFSwCodecWrapperBadBuffer);
   444 		}
   445 	ASSERT(iState==ERecordStateSendingBuffer || iState==ERecordStateSendingBufferPaused ||
   446 	       iState==ERecordStateSendingPartialBuffer || iState==ERecordStateSendingPartialBufferPaused ||
   447 	       iState==ERecordStateFailed);
   448 	switch (iState)
   449 		{
   450 		case ERecordStateSendingBuffer:
   451 		case ERecordStateSendingBufferPaused:
   452 			{
   453             iState = (iState==ERecordStateSendingBuffer) ? ERecordStateRecording : ERecordStateRecordingPaused;
   454 			if (!iInputHasFinished)
   455 			    {
   456 	            iAudioInput->BufferAck();
   457 			    }
   458 			break;
   459 			}
   460         case ERecordStateSendingPartialBuffer:
   461         case ERecordStateSendingPartialBufferPaused:
   462 			{
   463             iState = (iState==ERecordStateSendingPartialBuffer) ?
   464                 ERecordStateEmptiedPartialBuffer : ERecordStateEmptiedPartialBufferPaused;
   465 			RequestCallback(); // go back around to ensure next callback to client is asynchronous
   466 			break;
   467 			}
   468         default:
   469             {
   470             // anything else just ignore - e.g. are waiting for Stop following an error
   471             }
   472 		}
   473 #ifdef SYMBIAN_SCW_DEBUG
   474     RDebug::Print(_L("End CMMFSwcodecRecordDataPath::BufferEmptiedL() state=%s"), StateName(iState));
   475 #endif
   476 	}
   477 
   478 // Async callback support - used on PartialBuffer::BufferEmptiedL() transition
   479 
   480 void CMMFSwCodecRecordDataPath::RequestCallback()
   481 	{
   482 	iAsyncCallback->CallBack();
   483 	}
   484 
   485 TInt CMMFSwCodecRecordDataPath::Callback(TAny* aPtr)
   486 	{
   487 	CMMFSwCodecRecordDataPath* self = static_cast<CMMFSwCodecRecordDataPath*>(aPtr);
   488 	return self->DoCallback();
   489 	}
   490 
   491 TInt CMMFSwCodecRecordDataPath::DoCallback()
   492 	{
   493 	ASSERT(iState==ERecordStateEmptiedPartialBuffer || iState==ERecordStateEmptiedPartialBufferPaused); // only legal ones
   494 	TRAPD(err,ProcessBufferL(EFalse));
   495 	if (err != KErrNone)
   496 	    {
   497 	    iHwDeviceObserver->Error(err);
   498 	    }
   499 	return err;
   500 	}
   501 
   502 void CMMFSwCodecRecordDataPath::Stop()
   503 	{
   504 #ifdef SYMBIAN_SCW_DEBUG
   505     RDebug::Print(_L("CMMFSwcodecRecordDataPath::Stop() state=%s"), StateName(iState));
   506 #endif
   507 	iAudioInput->Close();
   508 	iState = ERecordStateCreated;
   509 #ifdef SYMBIAN_SCW_DEBUG
   510     RDebug::Print(_L("End CMMFSwcodecRecordDataPath::Stop() state=%s"), StateName(iState));
   511 #endif
   512 	}
   513 
   514 
   515 void CMMFSwCodecRecordDataPath::Pause()
   516 	{
   517 	// flush it anyway, whether we're active or not
   518 	// if we are active, then this should result in a call to RunL() pretty soon
   519 	//note that the Pause() in the context of record means buffers are
   520 	//continued to be obtained from the sound driver that have already
   521 	//been recorded - it just doesn't record any new audio data
   522 #ifdef SYMBIAN_SCW_DEBUG
   523 	RDebug::Print(_L("CMMFSwcodecRecordDataPath::Pause state=%s"), StateName(iState));
   524 #endif
   525 	switch (iState)
   526 	    {
   527 	    case ERecordStateRecording:
   528 	    case ERecordStateSendingBuffer:
   529 	    case ERecordStateSendingPartialBuffer:
   530 	    case ERecordStateEmptiedPartialBuffer:
   531 	        {
   532 	        iAudioInput->Pause();
   533 	        iState = KResumePauseTable[iState];
   534 	        }
   535 	        break;
   536 	    default: ;
   537 	        // do nothing - treat as no-op
   538 	    }
   539 #ifdef SYMBIAN_SCW_DEBUG
   540     RDebug::Print(_L("End CMMFSwcodecRecordDataPath::Pause state=%s"), StateName(iState));
   541 #endif
   542 	}
   543 
   544 
   545 RMdaDevSound& CMMFSwCodecRecordDataPath::Device()
   546 	{
   547 	ASSERT(EFalse); // TODO should not be called - future remove if we can
   548 	return iDummyDevSound;
   549 	}
   550 
   551 
   552 TUint CMMFSwCodecRecordDataPath::RecordedBytesCount()
   553 	{
   554 	return iRecordedBytesCount;
   555 	}
   556 
   557 /**
   558 Retrieves a custom interface to the device.
   559 The reference CMMFSwCodecWrapper supports one custom interfaces,
   560 MSetVbrFlagCustomInterface
   561 
   562 @param	aInterface
   563 		Interface UID, defined with the custom interface.
   564 		aInterface = KSetVbrFlagCustomInterfaceTypeUid for MSetVbrFlagCustomInterface.
   565 
   566 
   567 @return A pointer to the interface implementation, or NULL if the device can not
   568 		implement the interface requested. The return value must be cast to the
   569 		correct type by the user.
   570 */
   571 TAny* CMMFSwCodecRecordDataPath::CustomInterface(TUid aInterface)
   572 	{
   573 	TAny* ret = NULL;
   574 
   575 	if(aInterface.iUid == KSetVbrFlagCustomInterfaceTypeUid)
   576 		{
   577 		SetVbrFlag();
   578 		}
   579 	else if (aInterface == KUidSwSetParamInterface)
   580 		{
   581 		MSwSetParamInterface* self = this;
   582 		return self;
   583 		}
   584 	else if (aInterface == KUidSwInfoInterface)
   585 	    {
   586         MSwInfoInterface* self = this;
   587         return self;
   588 	    }
   589 	return ret;
   590 	}
   591 
   592 /**
   593 Used to set iVbrFlag on the datapath.
   594 
   595 This method is used to set the iVbrFlag in datapath. This flag is added to datapath to avail the
   596 alternative dataflow wherein datapath makes sure that destinationbuffer is filled to its maximum length
   597 before sending it to the sound driver. Sending the buffer directly to the device causes underflow incase of VBR codecs.
   598 */
   599 void CMMFSwCodecRecordDataPath::SetVbrFlag()
   600 	{
   601 	iVbrFlag = ETrue; // TODO this is seemingly redundant in a record case and could be pruned
   602 	}
   603 
   604 // MSwSetParamInterface - set various parameters etc
   605 
   606 TInt CMMFSwCodecRecordDataPath::SetSampleRate(TInt aSampleRate)
   607 	{
   608 	iSampleRate = aSampleRate;
   609 	return KErrNone;
   610 	}
   611 
   612 TInt CMMFSwCodecRecordDataPath::SetNumChannels(TInt aNumChannels)
   613 	{
   614 	iNumChannels = aNumChannels;
   615 	return KErrNone;
   616 	}
   617 
   618 TInt CMMFSwCodecRecordDataPath::SetGain(TInt aGain)
   619 	{
   620 	iGain = aGain; // cache here so would be used on next Initialize()
   621 	TInt error = KErrNone;
   622 	if (iAudioInput)
   623 		{
   624 		MAIParamInterface* paramInterface = static_cast<MAIParamInterface*>(iAudioInput->Interface(KUidAIParamInterface));
   625 		if (paramInterface)
   626 			{
   627 			error = paramInterface->SetGain(aGain);
   628 			}
   629 		}
   630 	return error;
   631 	}
   632 
   633 TInt CMMFSwCodecRecordDataPath::GetBufferSizes(TInt& aMinSize, TInt& aMaxSize)
   634 	{
   635 	TInt error = KErrNotReady;
   636 	if (iAudioInput)
   637 		{
   638 		MAIParamInterface* paramInterface = static_cast<MAIParamInterface*>(iAudioInput->Interface(KUidAIParamInterface));
   639 		if (paramInterface)
   640 			{
   641 			error = paramInterface->GetBufferSizes(aMinSize, aMaxSize);
   642 			}
   643 		}
   644 	return error;
   645 	}
   646 
   647 TInt CMMFSwCodecRecordDataPath::GetSupportedSampleRates(RArray<TInt>& aSupportedSampleRates)
   648     {
   649     TInt error = KErrNotReady;
   650     if (iAudioInput)
   651         {
   652         MAIParamInterface* paramInterface = static_cast<MAIParamInterface*>(iAudioInput->Interface(KUidAIParamInterface));
   653         if (paramInterface)
   654             {
   655             error = paramInterface->GetSupportedSampleRates(aSupportedSampleRates);
   656             }
   657         }
   658     return error;
   659     }
   660 
   661 CMMFSwCodecRecordDataPath::TSwCodecDataPathState CMMFSwCodecRecordDataPath::State() const
   662     {
   663     // Note: code assumes stopped, record and paused states are grouped consecutively
   664     if (iState==ERecordStateCreated)
   665         {
   666         return EStopped;
   667         }
   668     else if (iState >= ERecordStateRecording && iState <= ERecordStateEmptiedPartialBuffer)
   669         {
   670         return EPlaying;
   671         }
   672     else
   673         {
   674         return EPaused;
   675         }
   676     }
   677 
   678 TBool CMMFSwCodecRecordDataPath::RecordOrPause() const
   679     {
   680     // Note: code assumes stopped, record and paused states are grouped consecutively
   681     return iState >= ERecordStateRecording;
   682     }
   683 
   684 TBool CMMFSwCodecRecordDataPath::IsPaused() const
   685     {
   686     // Note: code assumes stopped, record and paused states are grouped consecutively
   687     return iState >= ERecordStateRecordingPaused;
   688     }
   689 
   690 // TODO - these functions are padding out from the old RMdaDevSound scheme.
   691 // They are no longer used here, but are used for playing...
   692 
   693 void CMMFSwCodecRecordDataPath::BufferFilledL(CMMFDataBuffer& /*aBuffer*/)
   694 	{
   695 	ASSERT(EFalse);
   696 	}
   697 
   698 void CMMFSwCodecRecordDataPath::SoundDeviceException(TInt /*aError*/)
   699 	{
   700 	ASSERT(EFalse);
   701 	}
   702