1.1 --- /dev/null Thu Jan 01 00:00:00 1970 +0000
1.2 +++ b/os/mm/devsound/devsoundrefplugin/src/swcodecwrapper/mmfSwCodecRecordDataPath.cpp Fri Jun 15 03:10:57 2012 +0200
1.3 @@ -0,0 +1,702 @@
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\mmfswcodecrecorddatapath.cpp
1.18 +//
1.19 +//
1.20 +
1.21 +#include "mmfSwCodecRecordDataPath.h"
1.22 +#include <mmf/server/mmfswcodecwrapper.h>
1.23 +#include <mmf/common/mmfpaniccodes.h>
1.24 +
1.25 +#ifdef SYMBIAN_SCW_DEBUG
1.26 +
1.27 +const TText* const KStateNames[] = // must agree with TRecordState
1.28 + {
1.29 + _S("ERecordStateCreated"),
1.30 + _S("ERecordStateFailed"),
1.31 + _S("ERecordStateRecording"),
1.32 + _S("ERecordStateSendingBuffer"),
1.33 + _S("ERecordStateSendingPartialBuffer"),
1.34 + _S("ERecordStateEmptiedPartialBuffer"),
1.35 + _S("ERecordStateRecordingPaused"),
1.36 + _S("ERecordStateSendingBufferPaused"),
1.37 + _S("ERecordStateSendingPartialBufferPaused"),
1.38 + _S("ERecordStateEmptiedPartialBufferPaused"),
1.39 + };
1.40 +
1.41 +static const TText* StateName(TInt aState)
1.42 + {
1.43 + return KStateNames[aState];
1.44 + }
1.45 +
1.46 +#endif // SYMBIAN_SCW_DEBUG
1.47 +
1.48 +// Table of next state used when resuming or pausing.
1.49 +const CMMFSwCodecRecordDataPath::TRecordState CMMFSwCodecRecordDataPath::KResumePauseTable[] =
1.50 + {
1.51 + ERecordStateCreated, //ERecordStateCreated // note order here is important - see State(), RecordOrPause() etc
1.52 + ERecordStateFailed, //ERecordStateFailed
1.53 + ERecordStateRecordingPaused, //ERecordStateRecording
1.54 + ERecordStateSendingBufferPaused, //ERecordStateSendingBuffer
1.55 + ERecordStateSendingPartialBufferPaused, //ERecordStateSendingPartialBuffer
1.56 + ERecordStateEmptiedPartialBufferPaused, //ERecordStateEmptiedPartialBuffer
1.57 + ERecordStateRecording, //ERecordStateRecordingPaused
1.58 + ERecordStateSendingBuffer, //ERecordStateSendingBufferPaused
1.59 + ERecordStateSendingPartialBuffer, //ERecordStateSendingPartialBufferPaused
1.60 + ERecordStateEmptiedPartialBuffer, //ERecordStateEmptiedPartialBufferPaused
1.61 + };
1.62 +
1.63 +
1.64 +CMMFSwCodecRecordDataPath* CMMFSwCodecRecordDataPath::NewL()
1.65 + {
1.66 + CMMFSwCodecRecordDataPath* self = new(ELeave) CMMFSwCodecRecordDataPath;
1.67 + CleanupStack::PushL(self);
1.68 + self->ConstructL();
1.69 + CleanupStack::Pop();
1.70 + return self;
1.71 + }
1.72 +
1.73 +
1.74 +void CMMFSwCodecRecordDataPath::ConstructL()
1.75 + {
1.76 + iAudioInput = MAudioInput::CreateL(*this);
1.77 + TCallBack callback(Callback, this);
1.78 + iAsyncCallback = new (ELeave) CAsyncCallBack(callback, CActive::EPriorityStandard);
1.79 + }
1.80 +
1.81 +CMMFSwCodecRecordDataPath::CMMFSwCodecRecordDataPath():
1.82 + iShadowData(NULL, 0, 0)
1.83 + {
1.84 + ASSERT(iState==ERecordStateCreated); // assume this value is 0, so no need to assign
1.85 + }
1.86 +
1.87 +CMMFSwCodecRecordDataPath::~CMMFSwCodecRecordDataPath()
1.88 + {
1.89 + if (iAudioInput)
1.90 + {
1.91 + iAudioInput->Release();
1.92 + }
1.93 +
1.94 + delete iCodecBuffer;
1.95 + delete iInputBuffer;
1.96 + delete iAsyncCallback;
1.97 + }
1.98 +
1.99 +
1.100 +TInt CMMFSwCodecRecordDataPath::SetObserver(MMMFHwDeviceObserver& aObserver)
1.101 + {
1.102 + TInt error;
1.103 + if (iHwDeviceObserver)
1.104 + {
1.105 + error = KErrAlreadyExists;
1.106 + }
1.107 + else
1.108 + {
1.109 + iHwDeviceObserver = &aObserver;
1.110 + error = KErrNone;
1.111 + }
1.112 + return error;
1.113 + }
1.114 +
1.115 +
1.116 +TInt CMMFSwCodecRecordDataPath::AddCodec(CMMFSwCodec& aCodec)
1.117 + {
1.118 + TInt err = KErrNone;
1.119 +
1.120 + if (iCodec)
1.121 + {
1.122 + err = KErrNotSupported; //doesn't support multiple codecs
1.123 + }
1.124 +
1.125 + if (!err)
1.126 + {
1.127 + iCodec = &aCodec;
1.128 +
1.129 + iSinkBufferSize = iCodec->SinkBufferSize();
1.130 + iAudioInputBufferSize = iCodec->SourceBufferSize(); // the buffer size we want from the input device
1.131 +
1.132 + if (!iSinkBufferSize || !iAudioInputBufferSize)
1.133 + {
1.134 + err = KErrArgument; //codec plugin has not specified buffer size
1.135 + }
1.136 + }
1.137 +
1.138 + if (!err)
1.139 + {
1.140 + // Allocate data buffer
1.141 + if (iCodec->IsNullCodec())
1.142 + {//don't need a separate sink buffer if null codec
1.143 + iSinkBuffer = NULL; //sink buffer is the sound device buffer
1.144 + iAudioInputBufferSize = iSinkBufferSize; // the audio input buffer becomes the sink buffer
1.145 + }
1.146 + else
1.147 + {//need a separate sink buffer for the codec - this is the buffer passed to our client
1.148 + ASSERT(!iCodecBuffer); // can't happen because can only call AddCodec once
1.149 + TRAP(err,iCodecBuffer = CMMFDataBuffer::NewL(iSinkBufferSize));
1.150 + iSinkBuffer = iCodecBuffer;
1.151 + }
1.152 + }
1.153 + if (!err)
1.154 + {
1.155 + ASSERT(!iInputBuffer); // can't happen because can only call AddCodec once
1.156 + TRAP(err,iInputBuffer = CMMFPtrBuffer::NewL());
1.157 + }
1.158 + if (!err)
1.159 + {
1.160 + // point iSinkBuffer at the right place
1.161 + if (iCodec->IsNullCodec())
1.162 + {
1.163 + iSinkBuffer = iInputBuffer;
1.164 + }
1.165 + else
1.166 + {
1.167 + iSinkBuffer = iCodecBuffer;
1.168 + }
1.169 + }
1.170 + return err;
1.171 + }
1.172 +
1.173 +
1.174 +TInt CMMFSwCodecRecordDataPath::Start()
1.175 + {
1.176 +#ifdef SYMBIAN_SCW_DEBUG
1.177 + RDebug::Print(_L("CMMFSwcodecRecordDataPath::Start() state=%s"), StateName(iState));
1.178 +#endif
1.179 + TInt err = KErrNone;
1.180 + if (!iCodec)
1.181 + {//check that a codec has been added
1.182 + err = KErrNotReady;
1.183 + }
1.184 + if (!err)
1.185 + {
1.186 + switch (iState)
1.187 + {
1.188 + case ERecordStateCreated:
1.189 + {
1.190 + TAudioInputParams params;
1.191 + params.iInitialGain = iGain;
1.192 + params.iSampleRate = iSampleRate;
1.193 + params.iNumChannels = iNumChannels;
1.194 + params.iNominalBufferSize = iAudioInputBufferSize;
1.195 + err = iAudioInput->Initialize(params);
1.196 + if (!err)
1.197 + {
1.198 + err = iAudioInput->Start();
1.199 + if (err)
1.200 + {
1.201 + iAudioInput->Close();
1.202 + ASSERT(iState == ERecordStateCreated); // end up in same state
1.203 + }
1.204 + else
1.205 + {
1.206 + iState = ERecordStateRecording;
1.207 + iInputHasFinished = EFalse;
1.208 + iRecordedBytesCount = 0; //used for debug purposes
1.209 + }
1.210 + }
1.211 + }
1.212 + break;
1.213 + case ERecordStateRecordingPaused:
1.214 + case ERecordStateSendingPartialBufferPaused:
1.215 + case ERecordStateEmptiedPartialBufferPaused:
1.216 + {
1.217 + // effectively in paused state, resume and switch to equivalent state
1.218 + iAudioInput->Resume();
1.219 + iInputHasFinished = EFalse;
1.220 + iState = KResumePauseTable[iState];
1.221 + }
1.222 + break;
1.223 + case ERecordStateSendingBufferPaused:
1.224 + {
1.225 + iAudioInput->Resume();
1.226 + if (iInputHasFinished)
1.227 + {
1.228 + iState = ERecordStateRecording; // as we follow InputHasFinished, we don't wait for the buffer
1.229 + iInputHasFinished = EFalse;
1.230 + }
1.231 + else
1.232 + {
1.233 + // effectively in paused state, resume and switch to equivalent state
1.234 + iState = KResumePauseTable[iState];
1.235 + }
1.236 + }
1.237 + break;
1.238 + case ERecordStateFailed:
1.239 + default:
1.240 + {
1.241 + // anything else assume already recording and ignore
1.242 + }
1.243 + break;
1.244 + }
1.245 + }
1.246 +
1.247 +#ifdef SYMBIAN_SCW_DEBUG
1.248 + RDebug::Print(_L("End CMMFSwcodecRecordDataPath::Start(%d) state=%s"), err, StateName(iState));
1.249 +#endif
1.250 + return err;
1.251 + }
1.252 +
1.253 +
1.254 +void CMMFSwCodecRecordDataPath::InputBufferAvailable(const TDesC8& aBuffer)
1.255 + {
1.256 +#ifdef SYMBIAN_SCW_DEBUG
1.257 + RDebug::Print(_L("CMMFSwcodecRecordDataPath::InputBufferAvailable(%d) state=%s"), aBuffer.Length(), StateName(iState));
1.258 +#endif
1.259 + ASSERT(iState==ERecordStateRecording || iState==ERecordStateRecordingPaused);
1.260 + iInputData = &aBuffer;
1.261 + TUint length = aBuffer.Length();
1.262 + // Update bytes recorded
1.263 + iRecordedBytesCount += length;
1.264 +
1.265 + //buffer ok can send to sink
1.266 + iInputOffset = 0;
1.267 + TRAPD(err,ProcessBufferL(EFalse)); //convert to sink data type using codec
1.268 + if (err != KErrNone)
1.269 + {
1.270 + iHwDeviceObserver->Error(err);
1.271 + }
1.272 +#ifdef SYMBIAN_SCW_DEBUG
1.273 + RDebug::Print(_L("End CMMFSwcodecRecordDataPath::InputBufferAvailable state=%s"), StateName(iState));
1.274 +#endif
1.275 + }
1.276 +
1.277 +void CMMFSwCodecRecordDataPath::InputFinished()
1.278 + {
1.279 +#ifdef SYMBIAN_SCW_DEBUG
1.280 + RDebug::Print(_L("CMMFSwcodecRecordDataPath::InputFinished state=%s"), StateName(iState));
1.281 +#endif
1.282 + ASSERT(iState==ERecordStateRecording || iState==ERecordStateRecordingPaused);
1.283 + iInputOffset = 0;
1.284 + iInputHasFinished = ETrue;
1.285 + TRAPD(err,ProcessBufferL(ETrue)); // finish off any conversion
1.286 + if (err != KErrNone)
1.287 + {
1.288 + iHwDeviceObserver->Error(err);
1.289 + }
1.290 +#ifdef SYMBIAN_SCW_DEBUG
1.291 + RDebug::Print(_L("End CMMFSwcodecRecordDataPath::InputFinished state=%s"), StateName(iState));
1.292 +#endif
1.293 + }
1.294 +
1.295 +void CMMFSwCodecRecordDataPath::InputError(TInt aError)
1.296 + {
1.297 +#ifdef SYMBIAN_SCW_DEBUG
1.298 + RDebug::Print(_L("CMMFSwcodecRecordDataPath::InputBufferAvailable(%d) state=%s"), aError, StateName(iState));
1.299 +#endif
1.300 + if (iState!=ERecordStateFailed)
1.301 + {
1.302 + iState = ERecordStateFailed;
1.303 + if (iHwDeviceObserver)
1.304 + {
1.305 + // Inform the observer of the exception condition
1.306 + // Assume it will subsequently call Stop(), and thus update policy
1.307 + iHwDeviceObserver->Error(aError);
1.308 + }
1.309 + }
1.310 +#ifdef SYMBIAN_SCW_DEBUG
1.311 + RDebug::Print(_L("CMMFSwcodecRecordDataPath::InputBufferAvailable() state=%s"), StateName(iState));
1.312 +#endif
1.313 + }
1.314 +
1.315 +
1.316 +/*
1.317 + * FillSinkBufferL
1.318 + *
1.319 + * Function to take the data from an already full source buffer and by using
1.320 + * a codec if necessary fills the sink buffer
1.321 + * If aLastBuffer, treat as a last buffer with zero length
1.322 + */
1.323 +void CMMFSwCodecRecordDataPath::ProcessBufferL(TBool aLastBuffer)
1.324 + {
1.325 + ASSERT(iState==ERecordStateRecording || iState==ERecordStateRecordingPaused ||
1.326 + iState==ERecordStateEmptiedPartialBuffer || iState==ERecordStateEmptiedPartialBufferPaused); // only valid states
1.327 + if (iCodec->IsNullCodec())
1.328 + {//no codec so sound device buffer can be used directly as sink buffer
1.329 + ASSERT(iSinkBuffer==iInputBuffer); // just assume this
1.330 + if (aLastBuffer)
1.331 + {
1.332 + iShadowData.Set(NULL, 0, 0);
1.333 + iInputBuffer->SetPtr(iShadowData);
1.334 + iInputBuffer->SetLastBuffer(ETrue);
1.335 + }
1.336 + else
1.337 + {
1.338 + iShadowData.Set(const_cast<TUint8*>(iInputData->Ptr()), iInputData->Length(), iInputData->Length());
1.339 + iInputBuffer->SetPtr(iShadowData);
1.340 + iInputBuffer->SetLastBuffer(EFalse);
1.341 + }
1.342 + iInputBuffer->SetStatus(EFull); //sink buffer is "full"
1.343 + TRecordState oldState = iState;
1.344 + switch (iState)
1.345 + {
1.346 + case ERecordStateRecording:
1.347 + iState = ERecordStateSendingBuffer;
1.348 + break;
1.349 + case ERecordStateRecordingPaused:
1.350 + iState = ERecordStateSendingBufferPaused;
1.351 + break;
1.352 + case ERecordStateEmptiedPartialBuffer:
1.353 + case ERecordStateEmptiedPartialBufferPaused:
1.354 + ASSERT(EFalse); // Technically these can occur but not if IsNullCodec is true, Complete is effectively always true
1.355 + break;
1.356 + }
1.357 + TInt err = iHwDeviceObserver->EmptyThisHwBuffer(*iSinkBuffer); //pass onto sink
1.358 + if (err)
1.359 + {
1.360 + iState = oldState; // rewind
1.361 + User::Leave(err);
1.362 + }
1.363 + }
1.364 + else
1.365 + {
1.366 + ASSERT(iSinkBuffer==iCodecBuffer); // sink and codec buffers are synonym, so just talk to iSinkBuffer
1.367 + if (aLastBuffer)
1.368 + {
1.369 + iShadowData.Set(NULL, 0, 0);
1.370 + iInputBuffer->SetPtr(iShadowData); // empty buffer
1.371 + iInputBuffer->SetLastBuffer(ETrue);
1.372 + }
1.373 + else
1.374 + {
1.375 + TPtrC8 tempData = iInputData->Mid(iInputOffset);
1.376 + iShadowData.Set(const_cast<TUint8*>(tempData.Ptr()), tempData.Length(), tempData.Length());
1.377 + iInputBuffer->SetPtr(iShadowData);
1.378 + iInputBuffer->SetLastBuffer(EFalse);
1.379 + }
1.380 + //pass buffer to codec for processing
1.381 + CMMFSwCodec::TCodecProcessResult codecProcessResult = iCodec->ProcessL(*iInputBuffer, *iSinkBuffer);
1.382 + if ((!iSinkBuffer->BufferSize())&&(codecProcessResult.iDstBytesAdded))
1.383 + {//the codec has added data but not set the buffer length
1.384 + iSinkBuffer->Data().SetLength(codecProcessResult.iDstBytesAdded);
1.385 + }
1.386 + //only supports EProcessComplete
1.387 + TRecordState oldState = iState;
1.388 + TInt err = KErrNone;
1.389 + switch (codecProcessResult.iCodecProcessStatus)
1.390 + {
1.391 + case CMMFSwCodec::TCodecProcessResult::EProcessComplete: //finished procesing source data - all data in sink buffer
1.392 + case CMMFSwCodec::TCodecProcessResult::EDstNotFilled: //finished procesing source data - sink buffer not full could be EOF
1.393 + case CMMFSwCodec::TCodecProcessResult::EEndOfData: //no more data - send what we've got to the sink
1.394 + {
1.395 + iSinkBuffer->SetStatus(EFull); // treat sink buffer as full
1.396 + iState = IsPaused() ? ERecordStateSendingBufferPaused : ERecordStateSendingBuffer;
1.397 + err = EmptyBufferL();
1.398 + break;
1.399 + }
1.400 + case CMMFSwCodec::TCodecProcessResult::EProcessIncomplete:
1.401 + {
1.402 + // codec has not yet finished with input - send buffer and come back around
1.403 + iSinkBuffer->SetStatus(EFull); // treat sink buffer as full
1.404 + iInputOffset = codecProcessResult.iSrcBytesProcessed;
1.405 + iState = IsPaused() ? ERecordStateSendingPartialBufferPaused : ERecordStateSendingPartialBuffer;
1.406 + err = EmptyBufferL();
1.407 + break;
1.408 + }
1.409 + default:
1.410 + Panic(EMMFSwCodecWrapperBadCodec); //should never get here - bad codec
1.411 + }
1.412 + if (err)
1.413 + {
1.414 + iState = oldState; // rewind prior to handle
1.415 + User::Leave(err);
1.416 + }
1.417 + }
1.418 + }
1.419 +
1.420 +TInt CMMFSwCodecRecordDataPath::EmptyBufferL()
1.421 + {
1.422 + // This code supports an assumption made by the vorbis encoder, which assumes it can safely generate empty buffers.
1.423 + // VbrFlag here implies the vorbis encoder.
1.424 + // TODO: Replace this with a generic solution - e.g. on EDstNotFilled we request a buffer from AudioInput instead
1.425 + // of calling the client back.
1.426 + if(iVbrFlag)
1.427 + {
1.428 + if(!iSinkBuffer->Data().Length() && !iInputBuffer->LastBuffer())
1.429 + {
1.430 + BufferEmptiedL(STATIC_CAST(CMMFDataBuffer&, *iSinkBuffer));
1.431 + return KErrNone;
1.432 + }
1.433 + }
1.434 + TInt err = iHwDeviceObserver->EmptyThisHwBuffer(*iSinkBuffer); //pass onto sink
1.435 + return err;
1.436 + }
1.437 +
1.438 +void CMMFSwCodecRecordDataPath::BufferEmptiedL(const CMMFDataBuffer& aBuffer)
1.439 + {
1.440 +#ifdef SYMBIAN_SCW_DEBUG
1.441 + RDebug::Print(_L("CMMFSwcodecRecordDataPath::BufferEmptiedL() state=%s"), StateName(iState));
1.442 +#endif
1.443 + if (&aBuffer != iSinkBuffer)
1.444 + {
1.445 + // we are only single buffering at the moment...
1.446 + Panic(EMMFSwCodecWrapperBadBuffer);
1.447 + }
1.448 + ASSERT(iState==ERecordStateSendingBuffer || iState==ERecordStateSendingBufferPaused ||
1.449 + iState==ERecordStateSendingPartialBuffer || iState==ERecordStateSendingPartialBufferPaused ||
1.450 + iState==ERecordStateFailed);
1.451 + switch (iState)
1.452 + {
1.453 + case ERecordStateSendingBuffer:
1.454 + case ERecordStateSendingBufferPaused:
1.455 + {
1.456 + iState = (iState==ERecordStateSendingBuffer) ? ERecordStateRecording : ERecordStateRecordingPaused;
1.457 + if (!iInputHasFinished)
1.458 + {
1.459 + iAudioInput->BufferAck();
1.460 + }
1.461 + break;
1.462 + }
1.463 + case ERecordStateSendingPartialBuffer:
1.464 + case ERecordStateSendingPartialBufferPaused:
1.465 + {
1.466 + iState = (iState==ERecordStateSendingPartialBuffer) ?
1.467 + ERecordStateEmptiedPartialBuffer : ERecordStateEmptiedPartialBufferPaused;
1.468 + RequestCallback(); // go back around to ensure next callback to client is asynchronous
1.469 + break;
1.470 + }
1.471 + default:
1.472 + {
1.473 + // anything else just ignore - e.g. are waiting for Stop following an error
1.474 + }
1.475 + }
1.476 +#ifdef SYMBIAN_SCW_DEBUG
1.477 + RDebug::Print(_L("End CMMFSwcodecRecordDataPath::BufferEmptiedL() state=%s"), StateName(iState));
1.478 +#endif
1.479 + }
1.480 +
1.481 +// Async callback support - used on PartialBuffer::BufferEmptiedL() transition
1.482 +
1.483 +void CMMFSwCodecRecordDataPath::RequestCallback()
1.484 + {
1.485 + iAsyncCallback->CallBack();
1.486 + }
1.487 +
1.488 +TInt CMMFSwCodecRecordDataPath::Callback(TAny* aPtr)
1.489 + {
1.490 + CMMFSwCodecRecordDataPath* self = static_cast<CMMFSwCodecRecordDataPath*>(aPtr);
1.491 + return self->DoCallback();
1.492 + }
1.493 +
1.494 +TInt CMMFSwCodecRecordDataPath::DoCallback()
1.495 + {
1.496 + ASSERT(iState==ERecordStateEmptiedPartialBuffer || iState==ERecordStateEmptiedPartialBufferPaused); // only legal ones
1.497 + TRAPD(err,ProcessBufferL(EFalse));
1.498 + if (err != KErrNone)
1.499 + {
1.500 + iHwDeviceObserver->Error(err);
1.501 + }
1.502 + return err;
1.503 + }
1.504 +
1.505 +void CMMFSwCodecRecordDataPath::Stop()
1.506 + {
1.507 +#ifdef SYMBIAN_SCW_DEBUG
1.508 + RDebug::Print(_L("CMMFSwcodecRecordDataPath::Stop() state=%s"), StateName(iState));
1.509 +#endif
1.510 + iAudioInput->Close();
1.511 + iState = ERecordStateCreated;
1.512 +#ifdef SYMBIAN_SCW_DEBUG
1.513 + RDebug::Print(_L("End CMMFSwcodecRecordDataPath::Stop() state=%s"), StateName(iState));
1.514 +#endif
1.515 + }
1.516 +
1.517 +
1.518 +void CMMFSwCodecRecordDataPath::Pause()
1.519 + {
1.520 + // flush it anyway, whether we're active or not
1.521 + // if we are active, then this should result in a call to RunL() pretty soon
1.522 + //note that the Pause() in the context of record means buffers are
1.523 + //continued to be obtained from the sound driver that have already
1.524 + //been recorded - it just doesn't record any new audio data
1.525 +#ifdef SYMBIAN_SCW_DEBUG
1.526 + RDebug::Print(_L("CMMFSwcodecRecordDataPath::Pause state=%s"), StateName(iState));
1.527 +#endif
1.528 + switch (iState)
1.529 + {
1.530 + case ERecordStateRecording:
1.531 + case ERecordStateSendingBuffer:
1.532 + case ERecordStateSendingPartialBuffer:
1.533 + case ERecordStateEmptiedPartialBuffer:
1.534 + {
1.535 + iAudioInput->Pause();
1.536 + iState = KResumePauseTable[iState];
1.537 + }
1.538 + break;
1.539 + default: ;
1.540 + // do nothing - treat as no-op
1.541 + }
1.542 +#ifdef SYMBIAN_SCW_DEBUG
1.543 + RDebug::Print(_L("End CMMFSwcodecRecordDataPath::Pause state=%s"), StateName(iState));
1.544 +#endif
1.545 + }
1.546 +
1.547 +
1.548 +RMdaDevSound& CMMFSwCodecRecordDataPath::Device()
1.549 + {
1.550 + ASSERT(EFalse); // TODO should not be called - future remove if we can
1.551 + return iDummyDevSound;
1.552 + }
1.553 +
1.554 +
1.555 +TUint CMMFSwCodecRecordDataPath::RecordedBytesCount()
1.556 + {
1.557 + return iRecordedBytesCount;
1.558 + }
1.559 +
1.560 +/**
1.561 +Retrieves a custom interface to the device.
1.562 +The reference CMMFSwCodecWrapper supports one custom interfaces,
1.563 +MSetVbrFlagCustomInterface
1.564 +
1.565 +@param aInterface
1.566 + Interface UID, defined with the custom interface.
1.567 + aInterface = KSetVbrFlagCustomInterfaceTypeUid for MSetVbrFlagCustomInterface.
1.568 +
1.569 +
1.570 +@return A pointer to the interface implementation, or NULL if the device can not
1.571 + implement the interface requested. The return value must be cast to the
1.572 + correct type by the user.
1.573 +*/
1.574 +TAny* CMMFSwCodecRecordDataPath::CustomInterface(TUid aInterface)
1.575 + {
1.576 + TAny* ret = NULL;
1.577 +
1.578 + if(aInterface.iUid == KSetVbrFlagCustomInterfaceTypeUid)
1.579 + {
1.580 + SetVbrFlag();
1.581 + }
1.582 + else if (aInterface == KUidSwSetParamInterface)
1.583 + {
1.584 + MSwSetParamInterface* self = this;
1.585 + return self;
1.586 + }
1.587 + else if (aInterface == KUidSwInfoInterface)
1.588 + {
1.589 + MSwInfoInterface* self = this;
1.590 + return self;
1.591 + }
1.592 + return ret;
1.593 + }
1.594 +
1.595 +/**
1.596 +Used to set iVbrFlag on the datapath.
1.597 +
1.598 +This method is used to set the iVbrFlag in datapath. This flag is added to datapath to avail the
1.599 +alternative dataflow wherein datapath makes sure that destinationbuffer is filled to its maximum length
1.600 +before sending it to the sound driver. Sending the buffer directly to the device causes underflow incase of VBR codecs.
1.601 +*/
1.602 +void CMMFSwCodecRecordDataPath::SetVbrFlag()
1.603 + {
1.604 + iVbrFlag = ETrue; // TODO this is seemingly redundant in a record case and could be pruned
1.605 + }
1.606 +
1.607 +// MSwSetParamInterface - set various parameters etc
1.608 +
1.609 +TInt CMMFSwCodecRecordDataPath::SetSampleRate(TInt aSampleRate)
1.610 + {
1.611 + iSampleRate = aSampleRate;
1.612 + return KErrNone;
1.613 + }
1.614 +
1.615 +TInt CMMFSwCodecRecordDataPath::SetNumChannels(TInt aNumChannels)
1.616 + {
1.617 + iNumChannels = aNumChannels;
1.618 + return KErrNone;
1.619 + }
1.620 +
1.621 +TInt CMMFSwCodecRecordDataPath::SetGain(TInt aGain)
1.622 + {
1.623 + iGain = aGain; // cache here so would be used on next Initialize()
1.624 + TInt error = KErrNone;
1.625 + if (iAudioInput)
1.626 + {
1.627 + MAIParamInterface* paramInterface = static_cast<MAIParamInterface*>(iAudioInput->Interface(KUidAIParamInterface));
1.628 + if (paramInterface)
1.629 + {
1.630 + error = paramInterface->SetGain(aGain);
1.631 + }
1.632 + }
1.633 + return error;
1.634 + }
1.635 +
1.636 +TInt CMMFSwCodecRecordDataPath::GetBufferSizes(TInt& aMinSize, TInt& aMaxSize)
1.637 + {
1.638 + TInt error = KErrNotReady;
1.639 + if (iAudioInput)
1.640 + {
1.641 + MAIParamInterface* paramInterface = static_cast<MAIParamInterface*>(iAudioInput->Interface(KUidAIParamInterface));
1.642 + if (paramInterface)
1.643 + {
1.644 + error = paramInterface->GetBufferSizes(aMinSize, aMaxSize);
1.645 + }
1.646 + }
1.647 + return error;
1.648 + }
1.649 +
1.650 +TInt CMMFSwCodecRecordDataPath::GetSupportedSampleRates(RArray<TInt>& aSupportedSampleRates)
1.651 + {
1.652 + TInt error = KErrNotReady;
1.653 + if (iAudioInput)
1.654 + {
1.655 + MAIParamInterface* paramInterface = static_cast<MAIParamInterface*>(iAudioInput->Interface(KUidAIParamInterface));
1.656 + if (paramInterface)
1.657 + {
1.658 + error = paramInterface->GetSupportedSampleRates(aSupportedSampleRates);
1.659 + }
1.660 + }
1.661 + return error;
1.662 + }
1.663 +
1.664 +CMMFSwCodecRecordDataPath::TSwCodecDataPathState CMMFSwCodecRecordDataPath::State() const
1.665 + {
1.666 + // Note: code assumes stopped, record and paused states are grouped consecutively
1.667 + if (iState==ERecordStateCreated)
1.668 + {
1.669 + return EStopped;
1.670 + }
1.671 + else if (iState >= ERecordStateRecording && iState <= ERecordStateEmptiedPartialBuffer)
1.672 + {
1.673 + return EPlaying;
1.674 + }
1.675 + else
1.676 + {
1.677 + return EPaused;
1.678 + }
1.679 + }
1.680 +
1.681 +TBool CMMFSwCodecRecordDataPath::RecordOrPause() const
1.682 + {
1.683 + // Note: code assumes stopped, record and paused states are grouped consecutively
1.684 + return iState >= ERecordStateRecording;
1.685 + }
1.686 +
1.687 +TBool CMMFSwCodecRecordDataPath::IsPaused() const
1.688 + {
1.689 + // Note: code assumes stopped, record and paused states are grouped consecutively
1.690 + return iState >= ERecordStateRecordingPaused;
1.691 + }
1.692 +
1.693 +// TODO - these functions are padding out from the old RMdaDevSound scheme.
1.694 +// They are no longer used here, but are used for playing...
1.695 +
1.696 +void CMMFSwCodecRecordDataPath::BufferFilledL(CMMFDataBuffer& /*aBuffer*/)
1.697 + {
1.698 + ASSERT(EFalse);
1.699 + }
1.700 +
1.701 +void CMMFSwCodecRecordDataPath::SoundDeviceException(TInt /*aError*/)
1.702 + {
1.703 + ASSERT(EFalse);
1.704 + }
1.705 +