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