os/mm/devsound/devsoundrefplugin/src/swcodecwrapper/mmfSwCodecConvertDataPath.cpp
1.1 --- /dev/null Thu Jan 01 00:00:00 1970 +0000
1.2 +++ b/os/mm/devsound/devsoundrefplugin/src/swcodecwrapper/mmfSwCodecConvertDataPath.cpp Fri Jun 15 03:10:57 2012 +0200
1.3 @@ -0,0 +1,383 @@
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\mmfswcodecconvertdatapath.cpp
1.18 +//
1.19 +//
1.20 +
1.21 +#include "mmfSwCodecConvertDataPath.h"
1.22 +#include <mmf/server/mmfswcodecwrapper.h>
1.23 +#include <mmf/common/mmfpaniccodes.h>
1.24 +
1.25 +
1.26 +CMMFSwCodecConvertDataPath* CMMFSwCodecConvertDataPath::NewL()
1.27 + {
1.28 + CMMFSwCodecConvertDataPath* self = new(ELeave) CMMFSwCodecConvertDataPath;
1.29 + CleanupStack::PushL(self);
1.30 + self->ConstructL();
1.31 + CleanupStack::Pop();
1.32 + return self;
1.33 + }
1.34 +
1.35 +
1.36 +void CMMFSwCodecConvertDataPath::ConstructL()
1.37 + {
1.38 + iDataPathConverter = new (ELeave) CDataPathConverter(*this,CActive::EPriorityUserInput);
1.39 + }
1.40 +
1.41 +
1.42 +CMMFSwCodecConvertDataPath::~CMMFSwCodecConvertDataPath()
1.43 + {
1.44 + delete iDataPathConverter;
1.45 + if (iCodec)
1.46 + {
1.47 + delete iSourceBuffer;
1.48 + if (!iCodec->IsNullCodec())
1.49 + delete iSinkBuffer;
1.50 + }
1.51 + }
1.52 +
1.53 +
1.54 +TInt CMMFSwCodecConvertDataPath::SetObserver(MMMFHwDeviceObserver& aObserver)
1.55 + {
1.56 + TInt error;
1.57 + if (iHwDeviceObserver)
1.58 + {
1.59 + error = KErrAlreadyExists;
1.60 + }
1.61 + else
1.62 + {
1.63 + iHwDeviceObserver = &aObserver;
1.64 + error = KErrNone;
1.65 + }
1.66 + return error;
1.67 + }
1.68 +
1.69 +
1.70 +TInt CMMFSwCodecConvertDataPath::AddCodec(CMMFSwCodec& aCodec)
1.71 + {
1.72 + if (iCodec)
1.73 + return KErrNotSupported; //doesn't support multiple codecs
1.74 +
1.75 + TInt err = KErrNone;
1.76 + iCodec = &aCodec;
1.77 +
1.78 + iSourceBufferSize = iCodec->SourceBufferSize();
1.79 + iSinkBufferSize = iCodec->SinkBufferSize();
1.80 +
1.81 + if ((!iSourceBufferSize)||(!iSinkBufferSize))
1.82 + err = KErrArgument; //codec plugin has not specified buffer size
1.83 +
1.84 + if (err == KErrNone)
1.85 + {
1.86 + TRAP(err,iSourceBuffer = CMMFDataBuffer::NewL(iSourceBufferSize));
1.87 + }
1.88 +
1.89 + if (err == KErrNone)
1.90 + {
1.91 + // Allocate data buffer
1.92 + if (iCodec->IsNullCodec())
1.93 + {//don't need a separate sink buffer if null codec
1.94 + iSinkBuffer = NULL; //sink buffer is the sound device buffer
1.95 + }
1.96 + else
1.97 + {//need a separate sink buffer for the codec
1.98 + TRAP(err,iSinkBuffer = CMMFDataBuffer::NewL(iSinkBufferSize));
1.99 + }
1.100 + }
1.101 + return err;
1.102 + }
1.103 +
1.104 +
1.105 +TInt CMMFSwCodecConvertDataPath::Start()
1.106 + {
1.107 + TInt startError = KErrNone;
1.108 + if (!iCodec)
1.109 + {//check that a codec has been added
1.110 + startError = KErrNotReady;
1.111 + }
1.112 + if (!startError)
1.113 + {
1.114 + // Start the player objects
1.115 + iSourceBuffer->SetLastBuffer(EFalse);
1.116 + iDataPathConverter->Start();
1.117 + iState = EPlaying;
1.118 + iNoMoreSourceData = EFalse;
1.119 + }
1.120 + return startError;
1.121 + }
1.122 +
1.123 +
1.124 +
1.125 +void CMMFSwCodecConvertDataPath::Stop()
1.126 + {
1.127 + iDataPathConverter->Cancel();
1.128 + iState = EStopped;
1.129 + }
1.130 +
1.131 +
1.132 +void CMMFSwCodecConvertDataPath::Pause()
1.133 + {//pause is equivalent to stop for a data transfer
1.134 + iDataPathConverter->Cancel();
1.135 + iState = EStopped;
1.136 + }
1.137 +
1.138 +/*** Main play loop ***/
1.139 +
1.140 +void CMMFSwCodecConvertDataPath::FillSourceBufferL()
1.141 + {
1.142 + STATIC_CAST(CMMFDataBuffer*, iSourceBuffer)->SetRequestSizeL(iSourceBufferSize);
1.143 +
1.144 + // Ask immediately for data from the observer
1.145 + User::LeaveIfError(iHwDeviceObserver->FillThisHwBuffer(*iSourceBuffer));
1.146 + }
1.147 +
1.148 + /**
1.149 + * BufferFilledL.
1.150 + * (from MDataSink)
1.151 + *
1.152 + * called by the CMMFDataPath's MDataSource when it has filled the buffer
1.153 + * @internalComponent
1.154 + * @param aBuffer
1.155 + *
1.156 + */
1.157 +void CMMFSwCodecConvertDataPath::BufferFilledL(CMMFDataBuffer& aBuffer)
1.158 + {
1.159 + iSourceBuffer = &aBuffer;
1.160 + iSourceBuffer->SetStatus(EFull);
1.161 +
1.162 + //need to check that the buffer size is not 0 -
1.163 + //if so assume we've reached the end of the data
1.164 + if (!iSourceBuffer->BufferSize())
1.165 + {//no buffer - could be end of source or could be that the source has no data
1.166 + //also need to check the sink buffer is available else there is still some
1.167 + //stuff to do before the sink buffer is freed
1.168 + if (iSinkBuffer->Status()==EAvailable)
1.169 + {
1.170 + iNoMoreSourceData = ETrue;
1.171 + }
1.172 + }
1.173 + else
1.174 + {
1.175 + if (iSourceBuffer->LastBuffer()) //also check last buffer flag
1.176 + iNoMoreSourceData = ETrue;
1.177 + iDataPathConverter->ChangeConvertState(CDataPathConverter::EFillingSinkBuffer);
1.178 + }
1.179 + }
1.180 +
1.181 +/*
1.182 + * FillSinkBufferL
1.183 + *
1.184 + * Function to take the data from an already full source buffer and by using
1.185 + * a codec if necessary fills the sink buffer
1.186 + * @internalComponent
1.187 + */
1.188 +void CMMFSwCodecConvertDataPath::FillSinkBufferL()
1.189 + {
1.190 + CMMFSwCodec::TCodecProcessResult codecProcessResult;
1.191 +
1.192 + if (iCodec->IsNullCodec())
1.193 + {//no codec so data can be sent direct to sink
1.194 + iSinkBuffer = iSourceBuffer;
1.195 + iSinkBuffer->SetStatus(EFull); //sink buffer is full
1.196 + }
1.197 + else
1.198 + {
1.199 + //pass buffer to codec for processing
1.200 + codecProcessResult = iCodec->ProcessL(*iSourceBuffer, *iSinkBuffer);
1.201 + if (iSourceBuffer->LastBuffer()) //if source is last buffer so is sink
1.202 + iSinkBuffer->SetLastBuffer(ETrue);
1.203 + if ((!iSinkBuffer->BufferSize())&&(codecProcessResult.iDstBytesAdded))
1.204 + {//the codec has added data but not set the buffer length
1.205 + iSinkBuffer->Data().SetLength(codecProcessResult.iDstBytesAdded);
1.206 + }
1.207 +
1.208 + //only supports EProcessComplete
1.209 + switch (codecProcessResult.iCodecProcessStatus)
1.210 + {
1.211 + case CMMFSwCodec::TCodecProcessResult::EProcessComplete:
1.212 + //finished procesing source data - all data in sink buffer
1.213 + {
1.214 + iSourceBuffer->SetStatus(EAvailable); //source buffer is now avaialble
1.215 + iSinkBuffer->SetStatus(EFull); //sink buffer is full
1.216 + }
1.217 + break;
1.218 + case CMMFSwCodec::TCodecProcessResult::EDstNotFilled:
1.219 + //could be the last buffer in which case dst might not get filled
1.220 + {
1.221 + iSourceBuffer->SetStatus(EAvailable); //source buffer is now avaialble
1.222 + iSinkBuffer->SetStatus(EFull); //sink buffer is full
1.223 + }
1.224 + break;
1.225 + case CMMFSwCodec::TCodecProcessResult::EEndOfData:
1.226 + //no more data - send what we've got to the sink
1.227 + //note we can't always rely on this - in many cases the codec will not know when
1.228 + //it has reached the end of data.
1.229 + {
1.230 + iSourceBuffer->SetStatus(EAvailable); //source buffer is now avaialble
1.231 + iSinkBuffer->SetStatus(EFull);//sink buffer may not really be 'full' but its as full as it going to get
1.232 + //doesn't matter if sink buffer is not full
1.233 + }
1.234 + break;
1.235 + default:
1.236 + Panic(EMMFSwCodecWrapperBadCodec); //should never get here - bad codec
1.237 + }
1.238 + }
1.239 + iDataPathConverter->ChangeConvertState(CDataPathConverter::EEmptyingSinkBuffer);
1.240 + }
1.241 +
1.242 +
1.243 +void CMMFSwCodecConvertDataPath::EmptySinkBufferL()
1.244 + {
1.245 + User::LeaveIfError(iHwDeviceObserver->EmptyThisHwBuffer(*iSinkBuffer));
1.246 + }
1.247 +
1.248 +
1.249 +void CMMFSwCodecConvertDataPath::BufferEmptiedL(const CMMFDataBuffer& aBuffer)
1.250 + {
1.251 + if (&aBuffer != iSinkBuffer)
1.252 + User::Leave(KErrArgument);
1.253 + if (!iNoMoreSourceData)
1.254 + iDataPathConverter->ChangeConvertState(CDataPathConverter::EFillingSourceBuffer);
1.255 + else //no more source data so signal EOF
1.256 + SoundDeviceException(KErrEof);
1.257 + }
1.258 +
1.259 +void CMMFSwCodecConvertDataPath::SoundDeviceException(TInt aError)
1.260 + {
1.261 + // Inform the observer of the exception condition
1.262 + iHwDeviceObserver->Error(aError);
1.263 +
1.264 + Stop();
1.265 +
1.266 + // Let the observer know we're fully stopped
1.267 + iHwDeviceObserver->Stopped();
1.268 + }
1.269 +
1.270 +RMdaDevSound& CMMFSwCodecConvertDataPath::Device()
1.271 + {
1.272 + return iDummyDevSound;//convert doesn't have a RMdaDevSound
1.273 + }
1.274 +
1.275 +/*** End of main play loop ***/
1.276 +
1.277 +
1.278 +
1.279 +
1.280 +/************************************************************************
1.281 + * CDataPathConverter
1.282 + * This class performs the main data transfer between the source and the sink
1.283 + * This is done in a separate class as opposed to CMMFSwCodecConvertDataPath
1.284 + * because the class needs to be an active object to avoid recursive call stacks
1.285 + * in cases where the source and sink are not active objects - which is
1.286 + * the case with descriptors. Making CMMFSwCodecConvertDataPath derive
1.287 + * from CActive is less desirable as it would involve multiple inheretence
1.288 + ************************************************************************/
1.289 +
1.290 +CMMFSwCodecConvertDataPath::CDataPathConverter::CDataPathConverter(CMMFSwCodecConvertDataPath& aParent, TInt aPriority)
1.291 +: CActive(aPriority), iParent(aParent)
1.292 + {
1.293 + CActiveScheduler::Add(this);
1.294 + iConvertState = EIdle;
1.295 + }
1.296 +
1.297 +
1.298 +CMMFSwCodecConvertDataPath::CDataPathConverter::~CDataPathConverter()
1.299 + {
1.300 + Cancel();
1.301 + }
1.302 +
1.303 +/**
1.304 + * Start
1.305 + *
1.306 + * Starts active scheduler 'play' loop
1.307 + * @internalComponent
1.308 + */
1.309 +void CMMFSwCodecConvertDataPath::CDataPathConverter::Start()
1.310 + {
1.311 + // If we're not already active, complete a request on ourselves to kick off the state machine
1.312 + if (iConvertState == EIdle)
1.313 + iConvertState = EFillingSourceBuffer;
1.314 + if (!IsActive())
1.315 + {
1.316 + TRequestStatus* stat = &iStatus;
1.317 + User::RequestComplete(stat, KErrNone);
1.318 + SetActive();
1.319 + }
1.320 + }
1.321 +
1.322 +
1.323 +void CMMFSwCodecConvertDataPath::CDataPathConverter::ChangeConvertState(TConvertState aNewConvertState)
1.324 + {
1.325 + TRequestStatus* stat = &iStatus;
1.326 + //change state
1.327 + iConvertState = aNewConvertState;
1.328 + if (!IsActive())
1.329 + {
1.330 + User::RequestComplete(stat, KErrNone);
1.331 + SetActive();
1.332 + }
1.333 + }
1.334 +
1.335 +/*** Main Convert Loop ***/
1.336 +
1.337 +void CMMFSwCodecConvertDataPath::CDataPathConverter::FillSourceBufferL()
1.338 + {
1.339 + iParent.FillSourceBufferL();
1.340 + }
1.341 +
1.342 +void CMMFSwCodecConvertDataPath::CDataPathConverter::FillSinkBufferL()
1.343 + {
1.344 + iParent.FillSinkBufferL();
1.345 + }
1.346 +
1.347 +void CMMFSwCodecConvertDataPath::CDataPathConverter::EmptySinkBufferL()
1.348 + {
1.349 + iParent.EmptySinkBufferL();
1.350 + }
1.351 +
1.352 +/*** End of main convert loop ***/
1.353 +
1.354 +
1.355 +void CMMFSwCodecConvertDataPath::CDataPathConverter::RunL()
1.356 + {
1.357 + switch (iConvertState)
1.358 + {
1.359 + case EFillingSourceBuffer:
1.360 + FillSourceBufferL();
1.361 + break;
1.362 + case EFillingSinkBuffer:
1.363 + FillSinkBufferL();
1.364 + break;
1.365 + case EEmptyingSinkBuffer:
1.366 + EmptySinkBufferL();
1.367 + break;
1.368 + case EIdle:
1.369 + break;
1.370 + }
1.371 + }
1.372 +
1.373 +void CMMFSwCodecConvertDataPath::CDataPathConverter::DoCancel()
1.374 + {
1.375 + //don't need to do anything as we don't have any async requests to other objects
1.376 + }
1.377 +
1.378 +TInt CMMFSwCodecConvertDataPath::CDataPathConverter::RunError(TInt aError)
1.379 + {
1.380 + iParent.SoundDeviceException(aError);
1.381 + return KErrNone;
1.382 + }
1.383 +
1.384 +
1.385 +
1.386 +