os/mm/devsound/devsoundrefplugin/src/swcodecwrapper/mmfSwCodecConvertDataPath.cpp
Update contrib.
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\mmfswcodecconvertdatapath.cpp
18 #include "mmfSwCodecConvertDataPath.h"
19 #include <mmf/server/mmfswcodecwrapper.h>
20 #include <mmf/common/mmfpaniccodes.h>
23 CMMFSwCodecConvertDataPath* CMMFSwCodecConvertDataPath::NewL()
25 CMMFSwCodecConvertDataPath* self = new(ELeave) CMMFSwCodecConvertDataPath;
26 CleanupStack::PushL(self);
33 void CMMFSwCodecConvertDataPath::ConstructL()
35 iDataPathConverter = new (ELeave) CDataPathConverter(*this,CActive::EPriorityUserInput);
39 CMMFSwCodecConvertDataPath::~CMMFSwCodecConvertDataPath()
41 delete iDataPathConverter;
45 if (!iCodec->IsNullCodec())
51 TInt CMMFSwCodecConvertDataPath::SetObserver(MMMFHwDeviceObserver& aObserver)
54 if (iHwDeviceObserver)
56 error = KErrAlreadyExists;
60 iHwDeviceObserver = &aObserver;
67 TInt CMMFSwCodecConvertDataPath::AddCodec(CMMFSwCodec& aCodec)
70 return KErrNotSupported; //doesn't support multiple codecs
75 iSourceBufferSize = iCodec->SourceBufferSize();
76 iSinkBufferSize = iCodec->SinkBufferSize();
78 if ((!iSourceBufferSize)||(!iSinkBufferSize))
79 err = KErrArgument; //codec plugin has not specified buffer size
83 TRAP(err,iSourceBuffer = CMMFDataBuffer::NewL(iSourceBufferSize));
88 // Allocate data buffer
89 if (iCodec->IsNullCodec())
90 {//don't need a separate sink buffer if null codec
91 iSinkBuffer = NULL; //sink buffer is the sound device buffer
94 {//need a separate sink buffer for the codec
95 TRAP(err,iSinkBuffer = CMMFDataBuffer::NewL(iSinkBufferSize));
102 TInt CMMFSwCodecConvertDataPath::Start()
104 TInt startError = KErrNone;
106 {//check that a codec has been added
107 startError = KErrNotReady;
111 // Start the player objects
112 iSourceBuffer->SetLastBuffer(EFalse);
113 iDataPathConverter->Start();
115 iNoMoreSourceData = EFalse;
122 void CMMFSwCodecConvertDataPath::Stop()
124 iDataPathConverter->Cancel();
129 void CMMFSwCodecConvertDataPath::Pause()
130 {//pause is equivalent to stop for a data transfer
131 iDataPathConverter->Cancel();
135 /*** Main play loop ***/
137 void CMMFSwCodecConvertDataPath::FillSourceBufferL()
139 STATIC_CAST(CMMFDataBuffer*, iSourceBuffer)->SetRequestSizeL(iSourceBufferSize);
141 // Ask immediately for data from the observer
142 User::LeaveIfError(iHwDeviceObserver->FillThisHwBuffer(*iSourceBuffer));
149 * called by the CMMFDataPath's MDataSource when it has filled the buffer
154 void CMMFSwCodecConvertDataPath::BufferFilledL(CMMFDataBuffer& aBuffer)
156 iSourceBuffer = &aBuffer;
157 iSourceBuffer->SetStatus(EFull);
159 //need to check that the buffer size is not 0 -
160 //if so assume we've reached the end of the data
161 if (!iSourceBuffer->BufferSize())
162 {//no buffer - could be end of source or could be that the source has no data
163 //also need to check the sink buffer is available else there is still some
164 //stuff to do before the sink buffer is freed
165 if (iSinkBuffer->Status()==EAvailable)
167 iNoMoreSourceData = ETrue;
172 if (iSourceBuffer->LastBuffer()) //also check last buffer flag
173 iNoMoreSourceData = ETrue;
174 iDataPathConverter->ChangeConvertState(CDataPathConverter::EFillingSinkBuffer);
181 * Function to take the data from an already full source buffer and by using
182 * a codec if necessary fills the sink buffer
185 void CMMFSwCodecConvertDataPath::FillSinkBufferL()
187 CMMFSwCodec::TCodecProcessResult codecProcessResult;
189 if (iCodec->IsNullCodec())
190 {//no codec so data can be sent direct to sink
191 iSinkBuffer = iSourceBuffer;
192 iSinkBuffer->SetStatus(EFull); //sink buffer is full
196 //pass buffer to codec for processing
197 codecProcessResult = iCodec->ProcessL(*iSourceBuffer, *iSinkBuffer);
198 if (iSourceBuffer->LastBuffer()) //if source is last buffer so is sink
199 iSinkBuffer->SetLastBuffer(ETrue);
200 if ((!iSinkBuffer->BufferSize())&&(codecProcessResult.iDstBytesAdded))
201 {//the codec has added data but not set the buffer length
202 iSinkBuffer->Data().SetLength(codecProcessResult.iDstBytesAdded);
205 //only supports EProcessComplete
206 switch (codecProcessResult.iCodecProcessStatus)
208 case CMMFSwCodec::TCodecProcessResult::EProcessComplete:
209 //finished procesing source data - all data in sink buffer
211 iSourceBuffer->SetStatus(EAvailable); //source buffer is now avaialble
212 iSinkBuffer->SetStatus(EFull); //sink buffer is full
215 case CMMFSwCodec::TCodecProcessResult::EDstNotFilled:
216 //could be the last buffer in which case dst might not get filled
218 iSourceBuffer->SetStatus(EAvailable); //source buffer is now avaialble
219 iSinkBuffer->SetStatus(EFull); //sink buffer is full
222 case CMMFSwCodec::TCodecProcessResult::EEndOfData:
223 //no more data - send what we've got to the sink
224 //note we can't always rely on this - in many cases the codec will not know when
225 //it has reached the end of data.
227 iSourceBuffer->SetStatus(EAvailable); //source buffer is now avaialble
228 iSinkBuffer->SetStatus(EFull);//sink buffer may not really be 'full' but its as full as it going to get
229 //doesn't matter if sink buffer is not full
233 Panic(EMMFSwCodecWrapperBadCodec); //should never get here - bad codec
236 iDataPathConverter->ChangeConvertState(CDataPathConverter::EEmptyingSinkBuffer);
240 void CMMFSwCodecConvertDataPath::EmptySinkBufferL()
242 User::LeaveIfError(iHwDeviceObserver->EmptyThisHwBuffer(*iSinkBuffer));
246 void CMMFSwCodecConvertDataPath::BufferEmptiedL(const CMMFDataBuffer& aBuffer)
248 if (&aBuffer != iSinkBuffer)
249 User::Leave(KErrArgument);
250 if (!iNoMoreSourceData)
251 iDataPathConverter->ChangeConvertState(CDataPathConverter::EFillingSourceBuffer);
252 else //no more source data so signal EOF
253 SoundDeviceException(KErrEof);
256 void CMMFSwCodecConvertDataPath::SoundDeviceException(TInt aError)
258 // Inform the observer of the exception condition
259 iHwDeviceObserver->Error(aError);
263 // Let the observer know we're fully stopped
264 iHwDeviceObserver->Stopped();
267 RMdaDevSound& CMMFSwCodecConvertDataPath::Device()
269 return iDummyDevSound;//convert doesn't have a RMdaDevSound
272 /*** End of main play loop ***/
277 /************************************************************************
279 * This class performs the main data transfer between the source and the sink
280 * This is done in a separate class as opposed to CMMFSwCodecConvertDataPath
281 * because the class needs to be an active object to avoid recursive call stacks
282 * in cases where the source and sink are not active objects - which is
283 * the case with descriptors. Making CMMFSwCodecConvertDataPath derive
284 * from CActive is less desirable as it would involve multiple inheretence
285 ************************************************************************/
287 CMMFSwCodecConvertDataPath::CDataPathConverter::CDataPathConverter(CMMFSwCodecConvertDataPath& aParent, TInt aPriority)
288 : CActive(aPriority), iParent(aParent)
290 CActiveScheduler::Add(this);
291 iConvertState = EIdle;
295 CMMFSwCodecConvertDataPath::CDataPathConverter::~CDataPathConverter()
303 * Starts active scheduler 'play' loop
306 void CMMFSwCodecConvertDataPath::CDataPathConverter::Start()
308 // If we're not already active, complete a request on ourselves to kick off the state machine
309 if (iConvertState == EIdle)
310 iConvertState = EFillingSourceBuffer;
313 TRequestStatus* stat = &iStatus;
314 User::RequestComplete(stat, KErrNone);
320 void CMMFSwCodecConvertDataPath::CDataPathConverter::ChangeConvertState(TConvertState aNewConvertState)
322 TRequestStatus* stat = &iStatus;
324 iConvertState = aNewConvertState;
327 User::RequestComplete(stat, KErrNone);
332 /*** Main Convert Loop ***/
334 void CMMFSwCodecConvertDataPath::CDataPathConverter::FillSourceBufferL()
336 iParent.FillSourceBufferL();
339 void CMMFSwCodecConvertDataPath::CDataPathConverter::FillSinkBufferL()
341 iParent.FillSinkBufferL();
344 void CMMFSwCodecConvertDataPath::CDataPathConverter::EmptySinkBufferL()
346 iParent.EmptySinkBufferL();
349 /*** End of main convert loop ***/
352 void CMMFSwCodecConvertDataPath::CDataPathConverter::RunL()
354 switch (iConvertState)
356 case EFillingSourceBuffer:
359 case EFillingSinkBuffer:
362 case EEmptyingSinkBuffer:
370 void CMMFSwCodecConvertDataPath::CDataPathConverter::DoCancel()
372 //don't need to do anything as we don't have any async requests to other objects
375 TInt CMMFSwCodecConvertDataPath::CDataPathConverter::RunError(TInt aError)
377 iParent.SoundDeviceException(aError);