1.1 --- /dev/null Thu Jan 01 00:00:00 1970 +0000
1.2 +++ b/os/mm/mmresourcemgmt/mmresctrl/src/mmrcserver/mmrcservercontroller.cpp Fri Jun 15 03:10:57 2012 +0200
1.3 @@ -0,0 +1,917 @@
1.4 +// Copyright (c) 2007-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 +//
1.18 +
1.19 +#include "mmrcservercontroller.h"
1.20 +#include "mmrcerrorcodes.h"
1.21 +#include "mmrcserversession.h"
1.22 +#include "mmrcserverinfo.h"
1.23 +#include "mmrcserverrulemanager.h"
1.24 +#include "mmrcadaptationproxy.h"
1.25 +#include "mlogicalchain.h"
1.26 +#include <a3f/audioprocessingunittypeuids.h>
1.27 +
1.28 +/**
1.29 +@internalComponent
1.30 +
1.31 +This class implements a timer.
1.32 +*/
1.33 +class CARNNotificationObject : public CActive
1.34 + {
1.35 +
1.36 +public:
1.37 + static CARNNotificationObject* NewL(const TInt aPriority, CMMRCServerInfo& aMMRCServerInfo);
1.38 + ~CARNNotificationObject();
1.39 + void After(TTimeIntervalMicroSeconds32 aInterval);
1.40 + void Initialize();
1.41 +
1.42 +protected:
1.43 + void RunL();
1.44 + void DoCancel();
1.45 +
1.46 +private:
1.47 + CARNNotificationObject(const TInt aPriority);
1.48 + void ConstructL(CMMRCServerInfo& aMMRCServerInfo);
1.49 +
1.50 +public:
1.51 + TInt iIndex;
1.52 +
1.53 +protected:
1.54 + RTimer iTimer;
1.55 + CMMRCServerInfo* iMMRCServerInfo;
1.56 + };
1.57 +
1.58 +//------------------------------------------------------------------------------------
1.59 +//
1.60 +//------------------------------------------------------------------------------------
1.61 +
1.62 +/**
1.63 + * Constructor
1.64 + */
1.65 +CMMRCServerController::CMMRCServerController()
1.66 + {
1.67 + TRACE_CREATE();
1.68 + DP_CONTEXT(----> CMMRCServerController::CMMRCServerController *CD1*, CtxDevSound, DPLOCAL);
1.69 + DP_IN();
1.70 +
1.71 + DP_OUT();
1.72 + }
1.73 +
1.74 +/**
1.75 + * Destructor
1.76 + */
1.77 +CMMRCServerController::~CMMRCServerController()
1.78 + {
1.79 + DP_CONTEXT(----> CMMRCServerController::~CMMRCServerController *CD1*, CtxDevSound, DPLOCAL);
1.80 + DP_IN();
1.81 +
1.82 + delete iMMRCServerInfo;
1.83 + delete iMMRCServerRuleManager;
1.84 + delete iARNNotificationObject;
1.85 +
1.86 + DP_OUT();
1.87 + }
1.88 +
1.89 +/**
1.90 + * Constructs, and returns a pointer to, a new CMMRCServerController object.
1.91 + * Leaves on failure.
1.92 + * @return CMMRCServerController* A pointer to newly created utlitly object.
1.93 + */
1.94 +CMMRCServerController* CMMRCServerController::NewL()
1.95 + {
1.96 + DP_STATIC_CONTEXT(----> CMMRCServerController::NewL *CD1*, CtxDevSound, DPLOCAL);
1.97 + DP_IN();
1.98 +
1.99 + CMMRCServerController* self = NewLC();
1.100 + CleanupStack::Pop(self);
1.101 +
1.102 + DP0_RET(self, "self = 0x%x");
1.103 + }
1.104 +
1.105 +/**
1.106 + * Constructs, leaves object on the cleanup stack, and returns a pointer
1.107 + * to, a new CMMRCServerController object.
1.108 + * Leaves on failure.
1.109 + * @return CMMRCServerController* A pointer to newly created utlitly object.
1.110 + */
1.111 +CMMRCServerController* CMMRCServerController::NewLC()
1.112 + {
1.113 + DP_STATIC_CONTEXT(----> CMMRCServerController::NewLC *CD1*, CtxDevSound, DPLOCAL);
1.114 + DP_IN();
1.115 +
1.116 + CMMRCServerController* self = new(ELeave) CMMRCServerController();
1.117 + CleanupStack::PushL(self);
1.118 + self->ConstructL();
1.119 +
1.120 + DP0_RET(self, "self = 0x%x");
1.121 + }
1.122 +
1.123 +/**
1.124 + * CMMRCServerController::ConstructL
1.125 + * Symbian 2nd phase constructor can leave.
1.126 + */
1.127 +void CMMRCServerController::ConstructL()
1.128 + {
1.129 + DP_CONTEXT(----> CMMRCServerController::ConstructL *CD1*, CtxDevSound, DPLOCAL);
1.130 + DP_IN();
1.131 +
1.132 + iMMRCServerInfo = CMMRCServerInfo::NewL(*this);
1.133 + iMMRCServerRuleManager = CMMRCServerRuleManager::NewL(*this);
1.134 + iARNNotificationObject = CARNNotificationObject::NewL(0, *iMMRCServerInfo); //Check the value
1.135 +
1.136 + DP_OUT();
1.137 + }
1.138 +
1.139 +/**
1.140 +* Process the request pointer by a specific session
1.141 +* @param CMMRCServerSession& aServerSession
1.142 +* @return KerrNone in case of success of the process
1.143 +*/
1.144 +TInt CMMRCServerController::EnqueueRequestL(CMMRCServerSession& aServerSession)
1.145 + {
1.146 + DP_CONTEXT(----> CMMRCServerController::ProcessRequestL *CD1*, CtxDevSound, DPLOCAL);
1.147 + DP_IN();
1.148 +
1.149 + //add the request to the queue
1.150 + iMMRCServerInfo->PushRequestL(aServerSession);
1.151 + TInt err = ProcessNextRequestL();
1.152 +
1.153 + DP0_RET(err, "err=%d");
1.154 + }
1.155 +
1.156 +/**
1.157 + * Process the first request with the highest priority from the FIFO queue
1.158 + * @return KerrNone in case of success of the process
1.159 + */
1.160 + TInt CMMRCServerController::ProcessNextRequestL()
1.161 + {
1.162 + DP_CONTEXT(--------> CMMRCServerController::ProcessNextRequestL *CD1*, CtxDevSound, DPLOCAL);
1.163 + DP_IN();
1.164 +
1.165 + TInt err = KErrNone;
1.166 +
1.167 + TUint64 contextId;
1.168 + TMMRCServerState serverState;
1.169 + iMMRCServerInfo->ServerState(serverState, contextId);
1.170 +#ifdef SYMBIAN_MULTIMEDIA_ENABLE_LOGGING
1.171 + RDebug::Print(_L("serverState = %d"), serverState);
1.172 + RDebug::Print(_L("size of message queue = %d"), iMMRCServerInfo->RequestsCount());
1.173 + RDebug::Print(_L("size of inprocess queue = %d"), iMMRCServerInfo->AllocatedResourceContextsCount());
1.174 + RDebug::Print(_L("size of paused queue = %d"), iMMRCServerInfo->AllocatedPausedContextsCount());
1.175 +#endif
1.176 + if(EMMRCServerStateIdle != serverState)
1.177 + {
1.178 + DP0_RET(KErrNone, "%d");
1.179 + }
1.180 +
1.181 + CMMRCServerSession* serverSession = iMMRCServerInfo->PopAndRemoveFirstRequest(); //PopAndRemoveRequestFirstHighestPriority();
1.182 + //Pop the next next commited message with the highest priority
1.183 + if( NULL == serverSession )
1.184 + {
1.185 + DP0_RET(KErrNone, "%d");
1.186 + }
1.187 +
1.188 + if (!serverSession->HasUserEnvironmentCapability())
1.189 + {
1.190 + /* Check to see if it's a record use case and in IDLE, Active, or PRIME.
1.191 + // in which case you should refuse the request for reasons
1.192 + KErrPermissionDenied
1.193 + */
1.194 + MLogicalChain* requestedChange = serverSession->LogicalChainRequestedStack();
1.195 + if ((requestedChange!=NULL) && (requestedChange->StreamState() > EInitialized)) // trying to play
1.196 + {
1.197 + TInt count = requestedChange->AudioProcessingUnitsCount();
1.198 + for (TInt i=0; i<count; ++i)
1.199 + {
1.200 + if (KUidAudioDeviceSource == requestedChange->AudioProcessingUnitUid(i))
1.201 + {
1.202 + serverSession->NotifyClientOfError(KErrPermissionDenied);
1.203 + // The following return value, doesn't actually make it back to the client
1.204 + // We do however Notify the client above using NotifyClientOfError()
1.205 + DP0_RET(KErrPermissionDenied, "%d");
1.206 + }
1.207 + }
1.208 + }
1.209 +
1.210 + }
1.211 +
1.212 + //Pop the last commited message in process = the current resource in use
1.213 + CMMRCServerSession* serverSessionAllocatedResource = iMMRCServerInfo->PopFirstAllocatedResourceContext();
1.214 + //If no resource locked...
1.215 + if( NULL == serverSessionAllocatedResource )
1.216 + {
1.217 +
1.218 + err = SendResourceRequest ( *serverSession );
1.219 + }
1.220 + else //If a resource has been allocated or is used, possibly preemption
1.221 + {
1.222 + TAudioContextId contextId = serverSession->ContextId( );
1.223 + TAudioContextId contextIdAllocatedResource = serverSessionAllocatedResource->ContextId( );
1.224 + #ifdef SYMBIAN_MULTIMEDIA_ENABLE_LOGGING
1.225 + RDebug::Print(_L("contextId = %d"), contextId.Id());
1.226 + RDebug::Print(_L("contextIdAllocatedResource = %d"), contextIdAllocatedResource.Id());
1.227 + #endif
1.228 +
1.229 + //in case of processing the same context, remove the last committed request from the queue
1.230 + // of in-process request to avoid preemption within the same context.
1.231 + if(contextId == contextIdAllocatedResource)
1.232 + {
1.233 + err = SendResourceRequest ( *serverSession );
1.234 + }
1.235 + else // maybe preemption
1.236 + {
1.237 + //Pop the rule
1.238 +
1.239 + MLogicalChain* logicalChainUsingResource = serverSessionAllocatedResource->LogicalChainLastCommited();
1.240 + MLogicalChain* logicalChainNew = serverSession->LogicalChainRequestedStack();
1.241 + //Should not/cannot be null!
1.242 + if( NULL==logicalChainUsingResource || NULL==logicalChainNew )
1.243 + {
1.244 + #ifdef _DEBUG
1.245 + RDebug::Print(_L("!!!!CMMRCServerController::ProcessNextRequestL - logicalChainUsingResource or logicalChainNew NULL"));
1.246 + ASSERT(0);
1.247 + #endif
1.248 + User::Leave(EMMRCErrorLogicalChainNull);
1.249 + }
1.250 +
1.251 + TAudioState audioStateAllocatedResource = logicalChainUsingResource->StreamState();
1.252 + TAudioState audioStateNew = logicalChainNew->StreamState ( );
1.253 + TMMRCRule rule = DecisionRule(audioStateAllocatedResource, audioStateNew);
1.254 + //Process according to the rule
1.255 + if( rule == EAlwaysProcess )
1.256 + {
1.257 + err = SendResourceRequest( *serverSession );
1.258 + }
1.259 + else // process only in case of higher priority
1.260 + {
1.261 + TInt currentClientPriority = logicalChainUsingResource->Priority();
1.262 + TInt newClientPriority = logicalChainNew->Priority();
1.263 + TBool currentClient_HasMMDD = serverSessionAllocatedResource->HasMultimediaCapability();
1.264 + TBool newClient_HasMMDD = serverSession->HasMultimediaCapability();
1.265 +
1.266 + // If new client has multimedia capability and current doesn't have,
1.267 + // priorities don't matter preempt the current client
1.268 +
1.269 + // But if both are equal competitors (both has MMDD capability or both hasn't)
1.270 + // and the newcomer has higher priority preempt too
1.271 + if ( (!currentClient_HasMMDD && newClient_HasMMDD) ||
1.272 + ( (currentClient_HasMMDD == newClient_HasMMDD) &&
1.273 + (newClientPriority > currentClientPriority) ) )
1.274 + {
1.275 + //The process is delayed. First we have to preempt the current inprocess resource
1.276 + // so puch back the request (its session)
1.277 + iMMRCServerInfo->PushRequestL(*serverSession);
1.278 + //
1.279 + //send a premption message to the last commited message session
1.280 + err = SendPreemptionRequest ( *serverSessionAllocatedResource );
1.281 + }
1.282 + else
1.283 + {
1.284 + // Deny the resource
1.285 + TAudioContextId iId = serverSession->ContextId( );
1.286 + TBool iIsOnPauseList = iMMRCServerInfo->IsOnPauseList(iId);
1.287 + if(!iIsOnPauseList)
1.288 + {
1.289 + iMMRCServerInfo->PushPausedContextL(*serverSession);
1.290 + }
1.291 +// nothing pending, deny resources, and notify client.
1.292 + serverSession->AdaptationRequestAcknowledgment(EMMRCAdaptationToServerResourceIncomplete, KErrInUse);
1.293 +// Server is back to idle.
1.294 + iMMRCServerInfo->SwitchServerState(EMMRCServerStateIdle, 0);
1.295 + }
1.296 + }
1.297 + }
1.298 + }
1.299 +
1.300 +#ifdef SYMBIAN_MULTIMEDIA_ENABLE_LOGGING
1.301 + if(err != KErrNone)
1.302 + {
1.303 + RDebug::Print(_L("!!!!CMMRCServerController::ProcessNextRequestL - error = %d"), err);
1.304 + }
1.305 +#endif
1.306 +
1.307 + DP0_RET(err, "err=%d");
1.308 + }
1.309 +
1.310 +/**
1.311 + * Pop the decision rule on the base of the current in-process request and a new committed request
1.312 + * @param TAudioState aAudioStateAllocatedResource
1.313 + * @param TAudioState aAudioStateRequiringProcess
1.314 + * @return TMMRCRule
1.315 + */
1.316 +TMMRCRule CMMRCServerController::DecisionRule(TAudioState aAudioStateAllocatedResource, TAudioState aAudioStateRequiringProcess)
1.317 + {
1.318 + DP_CONTEXT(----> CMMRCServerController::DecisionRule *CD1*, CtxDevSound, DPLOCAL);
1.319 + DP_IN();
1.320 +
1.321 + TMMRCRule rule = iMMRCServerRuleManager->DecisionRule(aAudioStateAllocatedResource, aAudioStateRequiringProcess);
1.322 +
1.323 + DP0_RET(rule, "rule=%d");
1.324 + }
1.325 +
1.326 +/**
1.327 + * Pop the action reason on the base of the last committed request and the current one
1.328 + * @param TAudioState aAudioStateLastCommitted
1.329 + * @param TAudioState aAudioStateRequested
1.330 + * @return TReason
1.331 + */
1.332 +TReason CMMRCServerController::ActionReason(TAudioState aAudioStateLastCommitted, TAudioState aAudioStateRequested)
1.333 + {
1.334 + DP_CONTEXT(----> CMMRCServerController::ActionReason *CD1*, CtxDevSound, DPLOCAL);
1.335 + DP_IN();
1.336 +
1.337 + TReason reason = iMMRCServerRuleManager->ActionReason(aAudioStateLastCommitted, aAudioStateRequested);
1.338 +
1.339 + DP0_RET(reason, "reason=%d");
1.340 + }
1.341 +
1.342 +/**
1.343 + * Send a resource request to the adaptation
1.344 + * @param MLogicalChain* aOldChain
1.345 + * @param MLogicalChain* aNewChain
1.346 + * @return KerrNone in case of success of the process
1.347 + */
1.348 +TInt CMMRCServerController::SendResourceRequest( CMMRCServerSession& aServerSession )
1.349 + {
1.350 + DP_CONTEXT(----> CMMRCServerController::SendResourceRequest *CD1*, CtxDevSound, DPLOCAL);
1.351 + DP_IN();
1.352 +
1.353 + iMMRCServerInfo->SwitchServerState( EMMRCServerStateWaitResourceResponseFromAdaptation, aServerSession.ContextId() );
1.354 +
1.355 +// if entering APR state, remove from Paused list.
1.356 + TRAPD(err, aServerSession.StartProcessL());
1.357 + {
1.358 + MLogicalChain* logicalChainNew = aServerSession.LogicalChainRequested();
1.359 + if (logicalChainNew)
1.360 + {
1.361 + TAudioState audioStateNew = logicalChainNew->StreamState();
1.362 + if (
1.363 + (audioStateNew == EActive ) ||
1.364 + (audioStateNew == EPrimed) ||
1.365 + (audioStateNew == EIdle)
1.366 +
1.367 + )
1.368 + {
1.369 + TInt contextId = aServerSession.ContextId( );
1.370 + iMMRCServerInfo->PopAndRemovePausedContext(contextId);
1.371 + }
1.372 + }
1.373 + }
1.374 + if(err == KErrNone)
1.375 + {
1.376 + //Send request
1.377 + if (!aServerSession.SendRequestToAdaptation(ENormalRequest))
1.378 + {
1.379 +// TBD fail at current task.
1.380 +// return server shutting down
1.381 + }
1.382 + }
1.383 +
1.384 +#ifdef SYMBIAN_MULTIMEDIA_ENABLE_LOGGING
1.385 + if(err != KErrNone)
1.386 + {
1.387 + RDebug::Print(_L("!!!!CMMRCServerController::SendResourceRequest - error = %d"), err);
1.388 + }
1.389 +#endif
1.390 +
1.391 + DP0_RET(err, "err=%d");
1.392 + }
1.393 +
1.394 +/**
1.395 + * Send a preemption request to the adaptation
1.396 + * @param MLogicalChain* aOldChain
1.397 + * @param MLogicalChain* aNewChain
1.398 + * @return KerrNone in case of success of the process
1.399 + */
1.400 +TInt CMMRCServerController::SendPreemptionRequest( CMMRCServerSession& aServerSession )
1.401 + {
1.402 + DP_CONTEXT(----> CMMRCServerController::SendPreemptionRequest *CD1*, CtxDevSound, DPLOCAL);
1.403 + DP_IN();
1.404 +
1.405 + iMMRCServerInfo->SwitchServerState( EMMRCServerStateWaitPreemptionResponse, aServerSession.ContextId() );
1.406 +
1.407 + //Update logical chains
1.408 + TRAPD(err, aServerSession.StartPreemptionL());
1.409 +
1.410 +
1.411 + if(err == KErrNone)
1.412 + {
1.413 + //Send request
1.414 + if (!aServerSession.SendRequestToAdaptation(EPreemptionRequest))
1.415 + {
1.416 + // TBD fail at current task.
1.417 + // return server shutting down/low memory where necessary
1.418 + }
1.419 + }
1.420 +
1.421 +#ifdef SYMBIAN_MULTIMEDIA_ENABLE_LOGGING
1.422 + if(err != KErrNone)
1.423 + {
1.424 + RDebug::Print(_L("!!!!CMMRCServerController::SendPreemptionRequest - error = %d"), err);
1.425 + }
1.426 +#endif
1.427 +
1.428 + DP0_RET(err, "err=%d");
1.429 + }
1.430 +
1.431 +/**
1.432 + * Callback from the adaptation
1.433 + * @param MLogicalChain* aOldChain
1.434 + * @param MLogicalChain* aNewChain
1.435 + * @param TMMRCAdaptationToServerRequestResults aResult
1.436 + */
1.437 +void CMMRCServerController::AdaptationRequestAcknowledgment(CMMRCServerSession& aServerSession, TMMRCAdaptationToServerRequestResults aResult, TInt /*aError*/)
1.438 + {
1.439 + DP_CONTEXT(--------> CMMRCServerController::AdaptationRequestAcknowledgment *CD1*, CtxDevSound, DPLOCAL);
1.440 + DP_IN();
1.441 +#ifdef SYMBIAN_MULTIMEDIA_ENABLE_LOGGING
1.442 + RDebug::Print(_L("size of message queue = %d"), iMMRCServerInfo->RequestsCount());
1.443 + RDebug::Print(_L("size of inprocess queue = %d"), iMMRCServerInfo->AllocatedResourceContextsCount());
1.444 + RDebug::Print(_L("size of paused queue = %d"), iMMRCServerInfo->AllocatedPausedContextsCount());
1.445 +#endif
1.446 +
1.447 + TUint64 contextId;
1.448 + TMMRCServerState serverState;
1.449 + TInt err = KErrNone;
1.450 + iMMRCServerInfo->ServerState(serverState, contextId);
1.451 + switch(serverState)
1.452 + {
1.453 + case EMMRCServerStateWaitPreemptionResponse: // here we process the response
1.454 + {
1.455 + switch(aResult)
1.456 + {
1.457 + case EMMRCAdaptationToServerResourceComplete:
1.458 + {
1.459 + //we remove the request in process from the queue
1.460 + CMMRCServerSession* serverSession = iMMRCServerInfo->PopAndRemoveFirstAllocatedResourceContext();
1.461 + if(serverSession)
1.462 + {
1.463 + TRAP(err, iMMRCServerInfo->PushPausedContextL(*serverSession));
1.464 + DP1(DLERR,"EPushPauseContextLeft %d", err);
1.465 + __ASSERT_DEBUG(err == KErrNone, Panic(EPushPauseContextLeft));
1.466 + }
1.467 + //Do not wait for a client acknowledgment
1.468 + iMMRCServerInfo->SwitchServerState(EMMRCServerStateIdle, 0);
1.469 + }
1.470 + break;
1.471 + case EMMRCAdaptationToServerResourceTimeout:
1.472 + default:
1.473 + {
1.474 + //jjpy TODO: What if the preemption request failed?!
1.475 + // That will be specified/implemented during the integration with the resource manager
1.476 + //we remove the request in process from the queue
1.477 + iMMRCServerInfo->PopAndRemoveFirstAllocatedResourceContext();
1.478 + }
1.479 + } // end switch(aResult)
1.480 + } //end case EMMRCServerStateWaitPreemptionResponse
1.481 + break;
1.482 + case EMMRCServerStateWaitResourceResponseFromAdaptation: // here we process the response
1.483 + {
1.484 + switch(aResult)
1.485 + {
1.486 + case EMMRCAdaptationToServerResourceComplete:
1.487 + {
1.488 + CMMRCServerSession* serverSessionAllocatedResource = iMMRCServerInfo->PopFirstAllocatedResourceContext();
1.489 + if( serverSessionAllocatedResource ) //If some previously some context has allocated a resource
1.490 + {
1.491 + //check if same context, if not it might be an error
1.492 + TAudioContextId contextId = aServerSession.ContextId( );
1.493 + TAudioContextId contextIdAllocatedResource = serverSessionAllocatedResource->ContextId( );
1.494 + #ifdef SYMBIAN_MULTIMEDIA_ENABLE_LOGGING
1.495 + RDebug::Print(_L("contextId = %d"), contextId.Id());
1.496 + RDebug::Print(_L("contextIdAllocatedResource = %d"), contextIdAllocatedResource.Id());
1.497 + #endif
1.498 + MLogicalChain* logicalChainNew = aServerSession.LogicalChainRequested();
1.499 + MLogicalChain* logicalChainOld = aServerSession.LogicalChainLastCommited();
1.500 + TAudioState audioStateNew = logicalChainNew->StreamState ( );
1.501 + if(contextId != contextIdAllocatedResource)
1.502 + {
1.503 + //it is ok if the rule said process always
1.504 + MLogicalChain* logicalChainUsingResource = serverSessionAllocatedResource->LogicalChainLastCommited();
1.505 + TAudioState audioStateAllocatedResource = logicalChainUsingResource->StreamState();
1.506 + TAudioState audioStateNew = logicalChainNew->StreamState ( );
1.507 + TMMRCRule rule = DecisionRule(audioStateAllocatedResource, audioStateNew);
1.508 + if( rule != EAlwaysProcess )
1.509 + {
1.510 + //not possible in theory... this is an error
1.511 + #ifdef SYMBIAN_MULTIMEDIA_ENABLE_LOGGING
1.512 + RDebug::Print(_L("!!!!CMMRCServerController::AdaptationRequestAcknowledgment - contextId allocating a resource and new request different. Should be the same!"), iMMRCServerInfo->AllocatedResourceContextsCount());
1.513 + #endif
1.514 + }
1.515 + }
1.516 + else //same context here, so check if the resource has been deallocated
1.517 + {
1.518 + // in case of stop etc. remove from in process request
1.519 + //now check the reason
1.520 + TAudioState audioStateOld = EUninitialized;
1.521 + if(logicalChainOld)
1.522 + {
1.523 + audioStateOld = logicalChainOld->StreamState();
1.524 + }
1.525 + TReason reason = ActionReason(audioStateOld, audioStateNew);
1.526 +
1.527 + if (reason == ELoad)
1.528 + {
1.529 + iARNNotificationObject->Cancel();
1.530 + iMMRCServerInfo->PopAndRemovePausedContext(contextId);
1.531 + }
1.532 + else if (reason == EUnload)
1.533 + {
1.534 + iARNNotificationObject->Initialize();
1.535 + iMMRCServerInfo->PopAndRemoveFirstAllocatedResourceContextByContextId(contextId);
1.536 + }
1.537 + }
1.538 + }
1.539 + else //no in process request locking a resource
1.540 + {
1.541 + MLogicalChain* logicalChainNew = aServerSession.LogicalChainRequested();
1.542 + MLogicalChain* logicalChainOld = aServerSession.LogicalChainLastCommited();
1.543 + TAudioState audioStateNew = logicalChainNew->StreamState ( );
1.544 + TAudioState audioStateOld = EUninitialized;
1.545 + if(logicalChainOld)
1.546 + {
1.547 + audioStateOld = logicalChainOld->StreamState();
1.548 + }
1.549 + TReason reason = ActionReason(audioStateOld, audioStateNew);
1.550 + if( reason == ELoad ||
1.551 + reason == EPrimeReason ||
1.552 + reason == EActivate )
1.553 + {
1.554 + TRAP(err, iMMRCServerInfo->PushAllocatedResourceContextL( aServerSession ));
1.555 + DP1(DLERR,"EPushAllocatedResourceContextFailed %d", err);
1.556 + __ASSERT_DEBUG(err == KErrNone, Panic(EPushAllocatedResourceContextFailed));
1.557 + }
1.558 + }
1.559 + iMMRCServerInfo->SwitchServerState(EMMRCServerStateIdle, 0);
1.560 + }
1.561 + break;
1.562 + case EMMRCAdaptationToServerResourceTimeout:
1.563 + default:
1.564 + {
1.565 + //Just switch the state
1.566 + iMMRCServerInfo->SwitchServerState(EMMRCServerStateIdle, 0);
1.567 + }
1.568 + break;
1.569 + } //end switch switch(aResult)
1.570 + } // end case EMMRCServerStateWaitResourceResponseFromAdaptation
1.571 + break;
1.572 + case EMMRCServerStateIdle: //not possible....
1.573 + default:
1.574 + break;
1.575 + }
1.576 +
1.577 + TRAP(err, ProcessNextRequestL());
1.578 + DP1(DLERR,"EProcessNextRequestLeft %d", err);
1.579 + __ASSERT_DEBUG(err == KErrNone, Panic(EProcessNextRequestLeft));
1.580 + DP_OUT();
1.581 + }
1.582 +
1.583 +/*
1.584 + *
1.585 + */
1.586 +void CMMRCServerController::NotifyPausedClientsResourceUnallocated()
1.587 + {
1.588 + DP_CONTEXT(----> CMMRCServerController::NotifyClientsResourceUnallocated *CD1*, CtxDevSound, DPLOCAL);
1.589 + DP_IN();
1.590 +
1.591 + CMMRCServerSession* serverSession = iMMRCServerInfo->PopAndRemoveFirstPausedContext( );
1.592 + while( serverSession )
1.593 + {
1.594 + serverSession->NotifyClientResourceUnallocated();
1.595 + serverSession = iMMRCServerInfo->PopAndRemoveFirstPausedContext( );
1.596 + }
1.597 +
1.598 + DP_OUT();
1.599 + }
1.600 +
1.601 +/*
1.602 + *
1.603 + */
1.604 +void CMMRCServerController::CloseSessionByContextId(TInt32 aContextId)
1.605 + {
1.606 + DP_CONTEXT(----> CMMRCServerController::CloseSessionByContextId *CD1*, CtxDevSound, DPLOCAL);
1.607 + DP_IN();
1.608 +
1.609 + while( iMMRCServerInfo->PopAndRemoveFirstAllocatedResourceContextByContextId( aContextId ) ){}
1.610 + while( iMMRCServerInfo->PopAndRemoveFirstRequestByContextId( aContextId ) ){}
1.611 + while( iMMRCServerInfo->PopAndRemovePausedContext( aContextId ) ){}
1.612 + while( iMMRCServerInfo->PopAndRemoveFirstAPRContextByContextId( aContextId ) ){}
1.613 + TUint64 contextId;
1.614 + TMMRCServerState serverState;
1.615 + iMMRCServerInfo->ServerState(serverState, contextId);
1.616 + if( contextId == aContextId)
1.617 + {
1.618 + iMMRCServerInfo->SwitchServerState(EMMRCServerStateIdle, 0);
1.619 + }
1.620 +#ifdef SYMBIAN_MULTIMEDIA_ENABLE_LOGGING
1.621 + RDebug::Print(_L("size of message queue = %d"), iMMRCServerInfo->RequestsCount());
1.622 + RDebug::Print(_L("size of inprocess queue = %d"), iMMRCServerInfo->AllocatedResourceContextsCount());
1.623 + RDebug::Print(_L("size of paused queue = %d"), iMMRCServerInfo->AllocatedPausedContextsCount());
1.624 + RDebug::Print(_L("size of APR queue = %d"), iMMRCServerInfo->AllocatedAPRContextsCount());
1.625 +#endif
1.626 +
1.627 + DP_OUT();
1.628 + }
1.629 +
1.630 +/*
1.631 + *
1.632 + */
1.633 +void CMMRCServerController::ServerState(TMMRCServerState& aServerState, TUint64& aContextId) const
1.634 + {
1.635 + iMMRCServerInfo->ServerState(aServerState, aContextId);
1.636 + }
1.637 +
1.638 +/**
1.639 +
1.640 +*/
1.641 +TInt CMMRCServerController::EnqueueRequestForAPRL(CMMRCServerSession& aServerSession)
1.642 + {
1.643 + DP_CONTEXT(----> CMMRCServerController::EnqueueRequestForAPRL *CD1*, CtxDevSound, DPLOCAL);
1.644 + DP_IN();
1.645 +
1.646 + //add the request to the queue
1.647 + TInt err = KErrNone;
1.648 + //if already there, should return KErrAlreadyExists
1.649 +
1.650 + if (iMMRCServerInfo->IsOnAPRList(aServerSession.ContextId()))
1.651 + {
1.652 + err = KErrAlreadyExists;
1.653 + }
1.654 + else
1.655 + {
1.656 + iMMRCServerInfo->PushAPRContextL(aServerSession);
1.657 + }
1.658 + DP0_RET(err, "err=%d");
1.659 + }
1.660 +
1.661 +/**
1.662 +
1.663 +*/
1.664 +TInt CMMRCServerController::RemoveRequestForAPR(CMMRCServerSession& aServerSession)
1.665 + {
1.666 + DP_CONTEXT(----> CMMRCServerController::RemoveRequestForAPR *CD1*, CtxDevSound, DPLOCAL);
1.667 + DP_IN();
1.668 +
1.669 + TInt err = KErrNone;
1.670 + TInt32 iContextId = aServerSession.ContextId();
1.671 + TBool iIsOnAPRList = iMMRCServerInfo->IsOnAPRList(iContextId);
1.672 + if(iIsOnAPRList)
1.673 + {
1.674 +
1.675 + // Its possible that the remove request has occured whilst clients are being notified that the resource is available
1.676 + // Check if a notification sequence is currently under way
1.677 + if ( iARNNotificationObject->IsActive() )
1.678 + {
1.679 +
1.680 + // Iterate through each paused client
1.681 + for ( TInt index = 0 ; index < iMMRCServerInfo->AllocatedPausedContextsCount() ; index++ )
1.682 + {
1.683 +
1.684 + // Find the client that has requested removal in the list
1.685 + TAudioContextId id = iMMRCServerInfo->PausedContextByIndex(index)->ContextId();
1.686 + if ( aServerSession.ContextId() == id )
1.687 + {
1.688 + // Object has already received a notification
1.689 + if ( index < iARNNotificationObject->iIndex )
1.690 + {
1.691 + // decrement the notification objects list index to account for the item that will be removed from the queue
1.692 + --iARNNotificationObject->iIndex;
1.693 + }
1.694 +
1.695 + // no action needs to be taken if
1.696 + // index >= iMMRCServerInfo.iIndex
1.697 +
1.698 + break;
1.699 + }
1.700 + }
1.701 +
1.702 + }
1.703 +
1.704 + iMMRCServerInfo->PopAndRemoveFirstAPRContextByContextId(iContextId);
1.705 + iMMRCServerInfo->PopAndRemovePausedContext(iContextId); //removing this session from the list of paused too.
1.706 + }
1.707 + else
1.708 + {
1.709 + err = KErrCancel;
1.710 + }
1.711 +
1.712 + DP0_RET(err, "err=%d");
1.713 + }
1.714 +
1.715 +/**
1.716 +
1.717 +*/
1.718 +TInt CMMRCServerController::WillResumePlay()
1.719 + {
1.720 + DP_CONTEXT(----> CMMRCServerController::WillResumePlay *CD1*, CtxDevSound, DPLOCAL);
1.721 + DP_IN();
1.722 +
1.723 + TInt err = KErrNone;
1.724 + iARNNotificationObject->Cancel();
1.725 +
1.726 +
1.727 + DP0_RET(err, "err=%d");
1.728 + }
1.729 +
1.730 +
1.731 +//From CActive
1.732 +/**
1.733 + * Constructor.
1.734 + */
1.735 +CARNNotificationObject::CARNNotificationObject(const TInt aPriority)
1.736 +:CActive(aPriority)
1.737 + {
1.738 + TRACE_CREATE();
1.739 + DP_CONTEXT(----> CARNNotificationObject::CARNNotificationObject *CD1*, CtxDevSound, DPLOCAL);
1.740 + DP_IN();
1.741 +
1.742 +
1.743 + DP_OUT();
1.744 + }
1.745 +
1.746 +
1.747 +/**
1.748 + * Destructor.
1.749 + */
1.750 +CARNNotificationObject::~CARNNotificationObject()
1.751 + {
1.752 + DP_CONTEXT(----> CARNNotificationObject::~CARNNotificationObject *CD1*, CtxDevSound, DPLOCAL);
1.753 + DP_IN();
1.754 +
1.755 + Cancel();
1.756 + iTimer.Close();
1.757 + delete iMMRCServerInfo;
1.758 +
1.759 + DP_OUT();
1.760 + }
1.761 +
1.762 +
1.763 +/**
1.764 + CARNNotificationObject::ConstructL
1.765 + Symbian 2nd phase constructor can leave.
1.766 + */
1.767 +void CARNNotificationObject::ConstructL(CMMRCServerInfo& aMMRCServerInfo)
1.768 + {
1.769 + DP_CONTEXT(----> CARNNotificationObject::ConstructL *CD1*, CtxDevSound, DPLOCAL);
1.770 + DP_IN();
1.771 +
1.772 + iMMRCServerInfo = &aMMRCServerInfo;
1.773 + CActiveScheduler::Add(this);
1.774 + User::LeaveIfError(iTimer.CreateLocal());
1.775 +
1.776 +
1.777 + DP_OUT();
1.778 + }
1.779 +
1.780 +// ---------------------------------------------------------------------------
1.781 +// CARNNotificationObject::NewL
1.782 +// ---------------------------------------------------------------------------
1.783 +CARNNotificationObject* CARNNotificationObject::NewL(const TInt aPriority, CMMRCServerInfo& aMMRCServerInfo)
1.784 + {
1.785 + DP_STATIC_CONTEXT(----> CARNNotificationObject::NewL *CD1*, CtxDevSound, DPLOCAL);
1.786 + DP_IN();
1.787 +
1.788 + CARNNotificationObject* self = new(ELeave)CARNNotificationObject(aPriority);
1.789 + CleanupStack::PushL(self);
1.790 + self->ConstructL(aMMRCServerInfo);
1.791 + CleanupStack::Pop(self);
1.792 +
1.793 + DP0_RET(self, "0x%x");
1.794 + }
1.795 +
1.796 +// ---------------------------------------------------------------------------
1.797 +// CARNNotificationObject::RunL
1.798 +// ---------------------------------------------------------------------------
1.799 +void CARNNotificationObject::RunL()
1.800 + {
1.801 + DP_CONTEXT(CARNNotificationObject::RunL *CD1*, CtxDevSound, DPLOCAL);
1.802 + DP_IN();
1.803 +
1.804 + #if defined(_DEBUG) && defined(__WINS__)
1.805 + const TTimeIntervalMicroSeconds32 KTimer = 1.8*1000000; // 1.8 seconds
1.806 + #else
1.807 + const TTimeIntervalMicroSeconds32 KTimer = 800000; // 0.8 seconds
1.808 + #endif
1.809 +
1.810 + TInt numPausedContext = iMMRCServerInfo->AllocatedPausedContextsCount();
1.811 +
1.812 + // The list may have been modified whilst the AO was asleep, need to check that:
1.813 + // 1) iIndex is still in range
1.814 + // 2) the object at that location should still receive a notification
1.815 +
1.816 + if ( (iIndex > -1) && (iIndex < numPausedContext) )
1.817 + {
1.818 + CMMRCServerSession* mmrcSession = iMMRCServerInfo->PausedContextByIndex(iIndex);
1.819 +
1.820 + TAudioContextId id = mmrcSession->ContextId( );
1.821 + TBool isOnAPRList = iMMRCServerInfo->IsOnAPRList(id);
1.822 + if(isOnAPRList)
1.823 + {
1.824 + mmrcSession->NotifyClientResourceUnallocated();
1.825 + }
1.826 +
1.827 +
1.828 + for(TInt aux = iIndex + 1; aux < numPausedContext ; aux++)
1.829 + {
1.830 + mmrcSession = iMMRCServerInfo->PausedContextByIndex(aux);
1.831 + id = mmrcSession->ContextId( );
1.832 + isOnAPRList = iMMRCServerInfo->IsOnAPRList(id);
1.833 + if(isOnAPRList)
1.834 + {
1.835 + iIndex = aux;
1.836 + After(KTimer);
1.837 + break;
1.838 + }
1.839 + }
1.840 + }
1.841 +
1.842 + else
1.843 + {
1.844 + iIndex = 0;
1.845 + }
1.846 +
1.847 + DP_OUT();
1.848 + }
1.849 +
1.850 +// ---------------------------------------------------------------------------
1.851 +// CARNNotificationObject::After
1.852 +// ---------------------------------------------------------------------------
1.853 +void CARNNotificationObject::After(TTimeIntervalMicroSeconds32 aInterval)
1.854 + {
1.855 + DP_CONTEXT(CARNNotificationObject::After *CD1*, CtxDevSound, DPLOCAL);
1.856 + DP_IN();
1.857 +
1.858 + Cancel();
1.859 + iTimer.After(iStatus,aInterval);
1.860 + SetActive();
1.861 +
1.862 + DP_OUT();
1.863 + }
1.864 +
1.865 +// ---------------------------------------------------------------------------
1.866 +// CARNNotificationObject::Initialize
1.867 +// ---------------------------------------------------------------------------
1.868 +void CARNNotificationObject::Initialize()
1.869 + {
1.870 + DP_CONTEXT(CARNNotificationObject::Initialize *CD1*, CtxDevSound, DPLOCAL);
1.871 + DP_IN();
1.872 +
1.873 + #if defined(_DEBUG) && defined(__WINS__)
1.874 + const TTimeIntervalMicroSeconds32 KTimer = 1.8*1000000; // 1.8 seconds
1.875 + #else
1.876 + const TTimeIntervalMicroSeconds32 KTimer = 800000; // 0.8 seconds
1.877 + #endif
1.878 +
1.879 + TInt numPausedContext = iMMRCServerInfo->AllocatedPausedContextsCount();
1.880 + iIndex = 0;
1.881 + CMMRCServerSession* mmrcSession = NULL;
1.882 +
1.883 + if(numPausedContext) // don't bother if zero.
1.884 + {
1.885 + for(TInt i(0); i<numPausedContext ; ++i)
1.886 + {
1.887 + mmrcSession = iMMRCServerInfo->PausedContextByIndex(i);
1.888 + TAudioContextId id = mmrcSession->ContextId( );
1.889 + TBool isOnAPRList = iMMRCServerInfo->IsOnAPRList(id);
1.890 + if(isOnAPRList)
1.891 + {
1.892 + iIndex = i;
1.893 + After(KTimer);
1.894 + break;
1.895 + }
1.896 + }
1.897 + }
1.898 +
1.899 + DP_OUT();
1.900 + }
1.901 +
1.902 +// ---------------------------------------------------------------------------
1.903 +// CMMRCServer::DoCancel
1.904 +// ---------------------------------------------------------------------------
1.905 +void CARNNotificationObject::DoCancel()
1.906 + {
1.907 + DP_CONTEXT(CARNNotificationObject::DoCancel *CD1*, CtxDevSound, DPLOCAL);
1.908 + DP_IN();
1.909 +
1.910 + iTimer.Cancel();
1.911 +
1.912 + DP_OUT();
1.913 + }
1.914 +
1.915 +
1.916 +void CMMRCServerController::Panic(TMMRCServerControllerPanicCodes aCode)
1.917 + {
1.918 + User::Panic(KMMRCServerControllerPanicCategory, aCode);
1.919 + }
1.920 +//EOF