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".
8 // Initial Contributors:
9 // Nokia Corporation - initial contribution.
14 // source\server\mmfswcodecrecorddatapath.cpp
18 #include "mmfSwCodecRecordDataPath.h"
19 #include <mmf/server/mmfswcodecwrapper.h>
20 #include <mmf/common/mmfpaniccodes.h>
22 #ifdef SYMBIAN_SCW_DEBUG
24 const TText* const KStateNames[] = // must agree with TRecordState
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"),
38 static const TText* StateName(TInt aState)
40 return KStateNames[aState];
43 #endif // SYMBIAN_SCW_DEBUG
45 // Table of next state used when resuming or pausing.
46 const CMMFSwCodecRecordDataPath::TRecordState CMMFSwCodecRecordDataPath::KResumePauseTable[] =
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
61 CMMFSwCodecRecordDataPath* CMMFSwCodecRecordDataPath::NewL()
63 CMMFSwCodecRecordDataPath* self = new(ELeave) CMMFSwCodecRecordDataPath;
64 CleanupStack::PushL(self);
71 void CMMFSwCodecRecordDataPath::ConstructL()
73 iAudioInput = MAudioInput::CreateL(*this);
74 TCallBack callback(Callback, this);
75 iAsyncCallback = new (ELeave) CAsyncCallBack(callback, CActive::EPriorityStandard);
78 CMMFSwCodecRecordDataPath::CMMFSwCodecRecordDataPath():
79 iShadowData(NULL, 0, 0)
81 ASSERT(iState==ERecordStateCreated); // assume this value is 0, so no need to assign
84 CMMFSwCodecRecordDataPath::~CMMFSwCodecRecordDataPath()
88 iAudioInput->Release();
93 delete iAsyncCallback;
97 TInt CMMFSwCodecRecordDataPath::SetObserver(MMMFHwDeviceObserver& aObserver)
100 if (iHwDeviceObserver)
102 error = KErrAlreadyExists;
106 iHwDeviceObserver = &aObserver;
113 TInt CMMFSwCodecRecordDataPath::AddCodec(CMMFSwCodec& aCodec)
119 err = KErrNotSupported; //doesn't support multiple codecs
126 iSinkBufferSize = iCodec->SinkBufferSize();
127 iAudioInputBufferSize = iCodec->SourceBufferSize(); // the buffer size we want from the input device
129 if (!iSinkBufferSize || !iAudioInputBufferSize)
131 err = KErrArgument; //codec plugin has not specified buffer size
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
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;
152 ASSERT(!iInputBuffer); // can't happen because can only call AddCodec once
153 TRAP(err,iInputBuffer = CMMFPtrBuffer::NewL());
157 // point iSinkBuffer at the right place
158 if (iCodec->IsNullCodec())
160 iSinkBuffer = iInputBuffer;
164 iSinkBuffer = iCodecBuffer;
171 TInt CMMFSwCodecRecordDataPath::Start()
173 #ifdef SYMBIAN_SCW_DEBUG
174 RDebug::Print(_L("CMMFSwcodecRecordDataPath::Start() state=%s"), StateName(iState));
178 {//check that a codec has been added
185 case ERecordStateCreated:
187 TAudioInputParams params;
188 params.iInitialGain = iGain;
189 params.iSampleRate = iSampleRate;
190 params.iNumChannels = iNumChannels;
191 params.iNominalBufferSize = iAudioInputBufferSize;
192 err = iAudioInput->Initialize(params);
195 err = iAudioInput->Start();
198 iAudioInput->Close();
199 ASSERT(iState == ERecordStateCreated); // end up in same state
203 iState = ERecordStateRecording;
204 iInputHasFinished = EFalse;
205 iRecordedBytesCount = 0; //used for debug purposes
210 case ERecordStateRecordingPaused:
211 case ERecordStateSendingPartialBufferPaused:
212 case ERecordStateEmptiedPartialBufferPaused:
214 // effectively in paused state, resume and switch to equivalent state
215 iAudioInput->Resume();
216 iInputHasFinished = EFalse;
217 iState = KResumePauseTable[iState];
220 case ERecordStateSendingBufferPaused:
222 iAudioInput->Resume();
223 if (iInputHasFinished)
225 iState = ERecordStateRecording; // as we follow InputHasFinished, we don't wait for the buffer
226 iInputHasFinished = EFalse;
230 // effectively in paused state, resume and switch to equivalent state
231 iState = KResumePauseTable[iState];
235 case ERecordStateFailed:
238 // anything else assume already recording and ignore
244 #ifdef SYMBIAN_SCW_DEBUG
245 RDebug::Print(_L("End CMMFSwcodecRecordDataPath::Start(%d) state=%s"), err, StateName(iState));
251 void CMMFSwCodecRecordDataPath::InputBufferAvailable(const TDesC8& aBuffer)
253 #ifdef SYMBIAN_SCW_DEBUG
254 RDebug::Print(_L("CMMFSwcodecRecordDataPath::InputBufferAvailable(%d) state=%s"), aBuffer.Length(), StateName(iState));
256 ASSERT(iState==ERecordStateRecording || iState==ERecordStateRecordingPaused);
257 iInputData = &aBuffer;
258 TUint length = aBuffer.Length();
259 // Update bytes recorded
260 iRecordedBytesCount += length;
262 //buffer ok can send to sink
264 TRAPD(err,ProcessBufferL(EFalse)); //convert to sink data type using codec
267 iHwDeviceObserver->Error(err);
269 #ifdef SYMBIAN_SCW_DEBUG
270 RDebug::Print(_L("End CMMFSwcodecRecordDataPath::InputBufferAvailable state=%s"), StateName(iState));
274 void CMMFSwCodecRecordDataPath::InputFinished()
276 #ifdef SYMBIAN_SCW_DEBUG
277 RDebug::Print(_L("CMMFSwcodecRecordDataPath::InputFinished state=%s"), StateName(iState));
279 ASSERT(iState==ERecordStateRecording || iState==ERecordStateRecordingPaused);
281 iInputHasFinished = ETrue;
282 TRAPD(err,ProcessBufferL(ETrue)); // finish off any conversion
285 iHwDeviceObserver->Error(err);
287 #ifdef SYMBIAN_SCW_DEBUG
288 RDebug::Print(_L("End CMMFSwcodecRecordDataPath::InputFinished state=%s"), StateName(iState));
292 void CMMFSwCodecRecordDataPath::InputError(TInt aError)
294 #ifdef SYMBIAN_SCW_DEBUG
295 RDebug::Print(_L("CMMFSwcodecRecordDataPath::InputBufferAvailable(%d) state=%s"), aError, StateName(iState));
297 if (iState!=ERecordStateFailed)
299 iState = ERecordStateFailed;
300 if (iHwDeviceObserver)
302 // Inform the observer of the exception condition
303 // Assume it will subsequently call Stop(), and thus update policy
304 iHwDeviceObserver->Error(aError);
307 #ifdef SYMBIAN_SCW_DEBUG
308 RDebug::Print(_L("CMMFSwcodecRecordDataPath::InputBufferAvailable() state=%s"), StateName(iState));
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
320 void CMMFSwCodecRecordDataPath::ProcessBufferL(TBool aLastBuffer)
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
329 iShadowData.Set(NULL, 0, 0);
330 iInputBuffer->SetPtr(iShadowData);
331 iInputBuffer->SetLastBuffer(ETrue);
335 iShadowData.Set(const_cast<TUint8*>(iInputData->Ptr()), iInputData->Length(), iInputData->Length());
336 iInputBuffer->SetPtr(iShadowData);
337 iInputBuffer->SetLastBuffer(EFalse);
339 iInputBuffer->SetStatus(EFull); //sink buffer is "full"
340 TRecordState oldState = iState;
343 case ERecordStateRecording:
344 iState = ERecordStateSendingBuffer;
346 case ERecordStateRecordingPaused:
347 iState = ERecordStateSendingBufferPaused;
349 case ERecordStateEmptiedPartialBuffer:
350 case ERecordStateEmptiedPartialBufferPaused:
351 ASSERT(EFalse); // Technically these can occur but not if IsNullCodec is true, Complete is effectively always true
354 TInt err = iHwDeviceObserver->EmptyThisHwBuffer(*iSinkBuffer); //pass onto sink
357 iState = oldState; // rewind
363 ASSERT(iSinkBuffer==iCodecBuffer); // sink and codec buffers are synonym, so just talk to iSinkBuffer
366 iShadowData.Set(NULL, 0, 0);
367 iInputBuffer->SetPtr(iShadowData); // empty buffer
368 iInputBuffer->SetLastBuffer(ETrue);
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);
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);
383 //only supports EProcessComplete
384 TRecordState oldState = iState;
386 switch (codecProcessResult.iCodecProcessStatus)
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
392 iSinkBuffer->SetStatus(EFull); // treat sink buffer as full
393 iState = IsPaused() ? ERecordStateSendingBufferPaused : ERecordStateSendingBuffer;
394 err = EmptyBufferL();
397 case CMMFSwCodec::TCodecProcessResult::EProcessIncomplete:
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();
407 Panic(EMMFSwCodecWrapperBadCodec); //should never get here - bad codec
411 iState = oldState; // rewind prior to handle
417 TInt CMMFSwCodecRecordDataPath::EmptyBufferL()
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.
425 if(!iSinkBuffer->Data().Length() && !iInputBuffer->LastBuffer())
427 BufferEmptiedL(STATIC_CAST(CMMFDataBuffer&, *iSinkBuffer));
431 TInt err = iHwDeviceObserver->EmptyThisHwBuffer(*iSinkBuffer); //pass onto sink
435 void CMMFSwCodecRecordDataPath::BufferEmptiedL(const CMMFDataBuffer& aBuffer)
437 #ifdef SYMBIAN_SCW_DEBUG
438 RDebug::Print(_L("CMMFSwcodecRecordDataPath::BufferEmptiedL() state=%s"), StateName(iState));
440 if (&aBuffer != iSinkBuffer)
442 // we are only single buffering at the moment...
443 Panic(EMMFSwCodecWrapperBadBuffer);
445 ASSERT(iState==ERecordStateSendingBuffer || iState==ERecordStateSendingBufferPaused ||
446 iState==ERecordStateSendingPartialBuffer || iState==ERecordStateSendingPartialBufferPaused ||
447 iState==ERecordStateFailed);
450 case ERecordStateSendingBuffer:
451 case ERecordStateSendingBufferPaused:
453 iState = (iState==ERecordStateSendingBuffer) ? ERecordStateRecording : ERecordStateRecordingPaused;
454 if (!iInputHasFinished)
456 iAudioInput->BufferAck();
460 case ERecordStateSendingPartialBuffer:
461 case ERecordStateSendingPartialBufferPaused:
463 iState = (iState==ERecordStateSendingPartialBuffer) ?
464 ERecordStateEmptiedPartialBuffer : ERecordStateEmptiedPartialBufferPaused;
465 RequestCallback(); // go back around to ensure next callback to client is asynchronous
470 // anything else just ignore - e.g. are waiting for Stop following an error
473 #ifdef SYMBIAN_SCW_DEBUG
474 RDebug::Print(_L("End CMMFSwcodecRecordDataPath::BufferEmptiedL() state=%s"), StateName(iState));
478 // Async callback support - used on PartialBuffer::BufferEmptiedL() transition
480 void CMMFSwCodecRecordDataPath::RequestCallback()
482 iAsyncCallback->CallBack();
485 TInt CMMFSwCodecRecordDataPath::Callback(TAny* aPtr)
487 CMMFSwCodecRecordDataPath* self = static_cast<CMMFSwCodecRecordDataPath*>(aPtr);
488 return self->DoCallback();
491 TInt CMMFSwCodecRecordDataPath::DoCallback()
493 ASSERT(iState==ERecordStateEmptiedPartialBuffer || iState==ERecordStateEmptiedPartialBufferPaused); // only legal ones
494 TRAPD(err,ProcessBufferL(EFalse));
497 iHwDeviceObserver->Error(err);
502 void CMMFSwCodecRecordDataPath::Stop()
504 #ifdef SYMBIAN_SCW_DEBUG
505 RDebug::Print(_L("CMMFSwcodecRecordDataPath::Stop() state=%s"), StateName(iState));
507 iAudioInput->Close();
508 iState = ERecordStateCreated;
509 #ifdef SYMBIAN_SCW_DEBUG
510 RDebug::Print(_L("End CMMFSwcodecRecordDataPath::Stop() state=%s"), StateName(iState));
515 void CMMFSwCodecRecordDataPath::Pause()
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));
527 case ERecordStateRecording:
528 case ERecordStateSendingBuffer:
529 case ERecordStateSendingPartialBuffer:
530 case ERecordStateEmptiedPartialBuffer:
532 iAudioInput->Pause();
533 iState = KResumePauseTable[iState];
537 // do nothing - treat as no-op
539 #ifdef SYMBIAN_SCW_DEBUG
540 RDebug::Print(_L("End CMMFSwcodecRecordDataPath::Pause state=%s"), StateName(iState));
545 RMdaDevSound& CMMFSwCodecRecordDataPath::Device()
547 ASSERT(EFalse); // TODO should not be called - future remove if we can
548 return iDummyDevSound;
552 TUint CMMFSwCodecRecordDataPath::RecordedBytesCount()
554 return iRecordedBytesCount;
558 Retrieves a custom interface to the device.
559 The reference CMMFSwCodecWrapper supports one custom interfaces,
560 MSetVbrFlagCustomInterface
563 Interface UID, defined with the custom interface.
564 aInterface = KSetVbrFlagCustomInterfaceTypeUid for MSetVbrFlagCustomInterface.
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.
571 TAny* CMMFSwCodecRecordDataPath::CustomInterface(TUid aInterface)
575 if(aInterface.iUid == KSetVbrFlagCustomInterfaceTypeUid)
579 else if (aInterface == KUidSwSetParamInterface)
581 MSwSetParamInterface* self = this;
584 else if (aInterface == KUidSwInfoInterface)
586 MSwInfoInterface* self = this;
593 Used to set iVbrFlag on the datapath.
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.
599 void CMMFSwCodecRecordDataPath::SetVbrFlag()
601 iVbrFlag = ETrue; // TODO this is seemingly redundant in a record case and could be pruned
604 // MSwSetParamInterface - set various parameters etc
606 TInt CMMFSwCodecRecordDataPath::SetSampleRate(TInt aSampleRate)
608 iSampleRate = aSampleRate;
612 TInt CMMFSwCodecRecordDataPath::SetNumChannels(TInt aNumChannels)
614 iNumChannels = aNumChannels;
618 TInt CMMFSwCodecRecordDataPath::SetGain(TInt aGain)
620 iGain = aGain; // cache here so would be used on next Initialize()
621 TInt error = KErrNone;
624 MAIParamInterface* paramInterface = static_cast<MAIParamInterface*>(iAudioInput->Interface(KUidAIParamInterface));
627 error = paramInterface->SetGain(aGain);
633 TInt CMMFSwCodecRecordDataPath::GetBufferSizes(TInt& aMinSize, TInt& aMaxSize)
635 TInt error = KErrNotReady;
638 MAIParamInterface* paramInterface = static_cast<MAIParamInterface*>(iAudioInput->Interface(KUidAIParamInterface));
641 error = paramInterface->GetBufferSizes(aMinSize, aMaxSize);
647 TInt CMMFSwCodecRecordDataPath::GetSupportedSampleRates(RArray<TInt>& aSupportedSampleRates)
649 TInt error = KErrNotReady;
652 MAIParamInterface* paramInterface = static_cast<MAIParamInterface*>(iAudioInput->Interface(KUidAIParamInterface));
655 error = paramInterface->GetSupportedSampleRates(aSupportedSampleRates);
661 CMMFSwCodecRecordDataPath::TSwCodecDataPathState CMMFSwCodecRecordDataPath::State() const
663 // Note: code assumes stopped, record and paused states are grouped consecutively
664 if (iState==ERecordStateCreated)
668 else if (iState >= ERecordStateRecording && iState <= ERecordStateEmptiedPartialBuffer)
678 TBool CMMFSwCodecRecordDataPath::RecordOrPause() const
680 // Note: code assumes stopped, record and paused states are grouped consecutively
681 return iState >= ERecordStateRecording;
684 TBool CMMFSwCodecRecordDataPath::IsPaused() const
686 // Note: code assumes stopped, record and paused states are grouped consecutively
687 return iState >= ERecordStateRecordingPaused;
690 // TODO - these functions are padding out from the old RMdaDevSound scheme.
691 // They are no longer used here, but are used for playing...
693 void CMMFSwCodecRecordDataPath::BufferFilledL(CMMFDataBuffer& /*aBuffer*/)
698 void CMMFSwCodecRecordDataPath::SoundDeviceException(TInt /*aError*/)