os/graphics/windowing/windowserver/nga/SERVER/SERVER.CPP
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
Update contrib.
     1 // Copyright (c) 1995-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".
     7 //
     8 // Initial Contributors:
     9 // Nokia Corporation - initial contribution.
    10 //
    11 // Contributors:
    12 //
    13 // Description:
    14 // Window server 'server' class
    15 // 
    16 //
    17 
    18 #include "server.h"
    19 #include "panics.h"
    20 #include "wstop.h"
    21 #include "EVENT.H"
    22 #include <bitdraw.h>
    23 #include <hal.h>
    24 #include "inifile.h"
    25 #include "wspluginmanager.h"
    26 #include "graphics/windowserverconstants.h"
    27 
    28 GLREF_D CDebugLogBase *wsDebugLog;
    29 
    30 const TUint KRangeCount = 1; 
    31 // We use a lot of 64 bit time calculations, but a periodic can only cope with signed 32 bit times
    32 // which gives them a limit of about 35 minutes.
    33 // Fortunately, our animtions are safe if redrawn early.  Every half an hour isn't going to hurt.
    34 const TTimeIntervalMicroSeconds KHalfHour = 30 * 60 * 1000 * 1000;
    35 
    36 const TInt KWsServRanges[KRangeCount] = 
    37 	{	
    38 	0
    39 	};
    40 
    41 const TUint8 KElementsIndex[KRangeCount] =
    42 	{
    43 	CPolicyServer::EAlwaysPass,		
    44 	};
    45 
    46 const CPolicyServer::TPolicyElement KPolicyElements[] = 
    47 	{ 
    48 	{_INIT_SECURITY_POLICY_C1(ECapabilityPowerMgmt), CPolicyServer::EFailClient}, 
    49 	{_INIT_SECURITY_POLICY_C1(ECapabilitySwEvent), CPolicyServer::EFailClient}, 
    50 	{_INIT_SECURITY_POLICY_C1(ECapabilityWriteDeviceData), CPolicyServer::EFailClient} 
    51 	};
    52 
    53 const CPolicyServer::TPolicy KWsServPolicy =
    54 	{
    55 	CPolicyServer::EAlwaysPass, 
    56 	KRangeCount,
    57 	KWsServRanges,
    58 	KElementsIndex,
    59 	KPolicyElements 	
    60 	};
    61  	
    62 // CWindowServer::CDefaultAnimationScheduler \\\\\\\\\\\\\\\\\\\\\\\\\\\
    63 
    64 class CWindowServer::CDefaultAnimationScheduler: public CBase, public MWsAnimationScheduler
    65 	{
    66 	// Associates a screen number with a CActiveSchedulerWait intance. This is used to
    67 	// achieve synchronous update completion on a specific screen.
    68 	class CScreenUpdateWait : public CActiveSchedulerWait
    69 		{
    70 		public:
    71 			CScreenUpdateWait(TInt aScreenNumber) : iScreenNumber (aScreenNumber) {}
    72 			TInt iScreenNumber;	
    73 		};
    74 	
    75 	struct TScreenUpdateDetails
    76 		{
    77 		CWindowServer::CDefaultAnimationScheduler* iScheduler;
    78 		TInt  iScreenNumber;
    79 		};
    80 	
    81 	struct TSchedule
    82 		{
    83 		MWsScreen* iScreen; // used as a unique index, searching with FindInUnsignedKeyOrder
    84 		TInt  iScreenNumber;
    85 		TBool iScheduled;
    86 		TTime iWhen;
    87 		TBool iRedraw;
    88 		};
    89 public:
    90 	enum TInactivityBehaviour
    91 		{
    92 		EStopAnimation,
    93 		EStopAllDrawing,
    94 		EIgnore,
    95 		};
    96 	class CScreenState;
    97 
    98 	CDefaultAnimationScheduler(MWsGraphicDrawerEnvironment& aEnv);
    99 	~CDefaultAnimationScheduler();
   100 	void ConstructL();		//LeaveScan:  member of nested class declaration
   101 	// implementing MWsAnimationScheduler
   102 	void ScheduleAnimation(MWsScreen& aScreen,const TTime& aWhen);
   103 	void UnscheduleAnimation(MWsScreen& aScreen);
   104 	void Invalidate(const TGraphicDrawerId& aId);
   105 	void OnInactive();
   106 	void OnActive();
   107 	void ScheduleRedraw(MWsScreen& aScreen,const TTime& aWhen);
   108 	void DoRedrawNow(MWsScreen& aScreen);
   109 	void DoRedrawNow(MWsScreen& aScreen, MWsAnimationScheduler::MScreenUpdateObserver& aObserver);
   110 	void ClearScreenUpdateObserver(const MWsAnimationScheduler::MScreenUpdateObserver& aObserver);
   111 private:
   112 	static TBool OnIdleCallBack(TAny* aAny);
   113 	void ScheduleUpdate (TInt aScreenNumber, TBool aForce);
   114 	void OnIdleCallBack(TBool aForce);
   115 	static TBool InvokeDueAnimation(TAny* aAny);
   116 	void RedrawAllInvalidatedRegions (TInt aScreen);
   117 	TSchedule* GetScheduledScreenUpdate(TInt aScreen);
   118 	void InvokeDueAnimation(TInt aScreen);
   119 	void ProcessUpdateCompletion (TInt aScreenNumber);
   120 	TInt ScreenNumber(MWsScreen& aScreen) const;
   121 	TTimeIntervalMicroSeconds GetDueDelta (TBool aForceRedraw, TSchedule* aScheduledUpdate);
   122 private:
   123 	RPointerArray<CScreenUpdateWait> iRedrawWaitLoop; 
   124 	RPointerArray<CScreenState> iScreenState;
   125 	MWsGraphicDrawerEnvironment& iEnv;
   126 	static const TInt64 KRedrawGrace;
   127 	static const TInt64 KAnimationGrace;
   128 	CAsyncCallBack* iIdleInitiator;
   129 	TBool iInRedrawNow;
   130 	RArray<TSchedule> iSchedule;
   131 	TBool iInactive;
   132 	TBool iInactiveDraws;
   133 	TBool iRedrawScheduled;
   134 	TInt64 iRedrawGracePeriod;
   135 	TInt64 iAnimationGracePeriod;
   136 	TInactivityBehaviour iInactivityBehaviour;
   137 	};
   138 
   139 class CWindowServer::CDefaultAnimationScheduler::CScreenState : public CActive
   140 	{
   141 public:
   142 	static CScreenState* NewL (CDefaultAnimationScheduler* aScheduler, TInt aScreenOrdinal);
   143 	~CScreenState();
   144 	void SetActive ();
   145 	
   146 	void WaitForRedraws(MWsAnimationScheduler::MScreenUpdateObserver& aObserver, TInt aNumRedraws);
   147 	void ClearScreenUpdateObserver(const MWsAnimationScheduler::MScreenUpdateObserver& aObserver);
   148 	CPeriodic* iUpdateOn;
   149 	TTime      iExpectedTickTime;
   150 	TScreenUpdateDetails iScreenUpdateDetails;
   151 	RArray<TGraphicDrawerId> iInvalidated;
   152 	TBool 		iInvalidateAll;
   153 	
   154 private:
   155 	CScreenState (CDefaultAnimationScheduler* aScheduler, TInt aScreenOrdinal);
   156 	void ConstructL ();
   157 	void ReleaseRemainingClients();
   158 	void ReleaseClientsWaitingFor(TUint aCurrentFrame);
   159 	
   160 	void RunL();
   161 	void DoCancel() 
   162 		{
   163 		TRequestStatus* tmpTRS = &iStatus;
   164 		User::RequestComplete(tmpTRS, KErrNone);
   165 		};
   166 	class TWaitingClient
   167 		{
   168 	public:
   169 		TWaitingClient(MWsAnimationScheduler::MScreenUpdateObserver& aObserver, TInt aTargetFrame)
   170 			: iObserver(aObserver), iTargetFrame(aTargetFrame)
   171 			{
   172 			}
   173 		MWsAnimationScheduler::MScreenUpdateObserver& iObserver;
   174 		TUint iTargetFrame;
   175 		};
   176 	TUint iFrameCount;
   177 	RArray<TWaitingClient> iWaitingClients;
   178 	};
   179 
   180 // If using the default animation scheduler on a device, these two numbers may be worth tweaking in the inifile
   181 // However, both are maximum periods - wserv will go faster than either if nothing else is using the system.
   182 const TInt64 CWindowServer::CDefaultAnimationScheduler::KRedrawGrace = 0; // do redraws immediately
   183 const TInt64 CWindowServer::CDefaultAnimationScheduler::KAnimationGrace = 0; // do animation redraws immediately
   184 	
   185 CWindowServer::CDefaultAnimationScheduler::CDefaultAnimationScheduler(MWsGraphicDrawerEnvironment& aEnv):
   186 	iEnv(aEnv), iSchedule(1,_FOFF(TSchedule,iScreen))
   187 	{
   188 	}
   189 
   190 CWindowServer::CDefaultAnimationScheduler::~CDefaultAnimationScheduler()
   191 	{
   192 	iSchedule.Close();
   193 	delete iIdleInitiator;
   194 	
   195 	TInt screenCount = iScreenState.Count();
   196 	for (TInt ii = 0; ii < screenCount; ii++)
   197 		delete iScreenState[ii];
   198 
   199 	iScreenState.Close();
   200 
   201 	// Destroy the redraw wait loops associated each screen, and close
   202 	// the associated Array objects.
   203 	TInt waitLoopCount = iRedrawWaitLoop.Count();
   204 	for (TInt waitLoop = 0; waitLoop < waitLoopCount; waitLoop++)
   205 		delete iRedrawWaitLoop[waitLoop];
   206 	iRedrawWaitLoop.Close();
   207 	}
   208 	
   209 void CWindowServer::CDefaultAnimationScheduler::ConstructL()
   210 	{
   211 	_LIT(KOnInactive,"ONINACTIVE");
   212 	_LIT(KStopAnimation,"STOPANIMATION");
   213 	_LIT(KStopAllDrawing,"STOPALLDRAWING");
   214 	_LIT(KIgnore,"IGNORE");
   215 	
   216 	TPtrC inactivityBehaviourString;
   217 	WsIniFile->FindVar(KOnInactive,inactivityBehaviourString);
   218 	if(inactivityBehaviourString.CompareF(KStopAnimation)==0)
   219 		iInactivityBehaviour = EStopAnimation;
   220 	else if(inactivityBehaviourString.CompareF(KStopAllDrawing)==0)
   221 		iInactivityBehaviour = EStopAllDrawing;
   222 	else if(inactivityBehaviourString.CompareF(KIgnore)==0)
   223 		iInactivityBehaviour = EIgnore;
   224 		
   225 	_LIT(KRedrawGracePeriod, "REDRAWGRACEPERIOD");
   226 	TInt tmp = KRedrawGrace;
   227 	WsIniFile->FindVar(KRedrawGracePeriod, tmp);
   228 	iRedrawGracePeriod = tmp;
   229 	
   230 	_LIT(KAnimationGracePeriod, "ANIMATIONGRACEPERIOD");
   231 	tmp = KAnimationGrace;
   232 	WsIniFile->FindVar(KAnimationGracePeriod, tmp);
   233 	iAnimationGracePeriod = tmp;
   234 	
   235 	iIdleInitiator = new(ELeave) CAsyncCallBack(TCallBack(OnIdleCallBack,this),EWsGraphicAnimateAwaitIdlePriority);
   236 	
   237 	TInt screenCount;
   238 	User::LeaveIfError(HAL::Get( HAL::EDisplayNumberOfScreens, screenCount)); 
   239 	for (TInt i = 0; i < screenCount; i++)
   240 		{
   241 		CScreenState* screenState = CScreenState::NewL (this, i);
   242 		CleanupStack::PushL(screenState);
   243 		iScreenState.AppendL (screenState);
   244 		CleanupStack::Pop(screenState);
   245 		}
   246 		
   247 	
   248 	// Ensure that the wait loop array has some allocated slots, making it highly 
   249 	// unlikely that Append() will fail due to OOM.
   250 	iRedrawWaitLoop.ReserveL (8);
   251 	}
   252 
   253 void CWindowServer::CDefaultAnimationScheduler::Invalidate(const TGraphicDrawerId& aId)
   254 	{
   255 	const TInt screenCount = iEnv.ScreenCount();
   256 	for(TInt ii = 0; ii < screenCount; ii++)
   257 		{
   258 		CScreenState* screenState = iScreenState[ii];
   259 		if(!screenState->iInvalidateAll)
   260 			{
   261 			switch(screenState->iInvalidated.InsertInOrder(aId,TLinearOrder<TGraphicDrawerId>(TGraphicDrawerId::Compare)))
   262 				{
   263 				case KErrNone:
   264 				case KErrAlreadyExists:
   265 					break;
   266 				default:
   267 					screenState->iInvalidateAll = ETrue;
   268 					screenState->iInvalidated.Reset();
   269 				}
   270 			}
   271 		}
   272 	iIdleInitiator->CallBack();
   273 	}
   274 
   275 CWindowServer::CDefaultAnimationScheduler::CScreenState::CScreenState (CDefaultAnimationScheduler* aScheduler, TInt aScreenNumber):
   276 	CActive(EComposeCompletePriority)
   277 	{
   278 	iScreenUpdateDetails.iScheduler = aScheduler;
   279 	iScreenUpdateDetails.iScreenNumber = aScreenNumber;
   280 	}
   281 
   282 CWindowServer::CDefaultAnimationScheduler::CScreenState* 
   283 CWindowServer::CDefaultAnimationScheduler::CScreenState::NewL (CDefaultAnimationScheduler* aScheduler, TInt aScreenOrdinal)
   284 	{
   285 	CScreenState* self = new(ELeave)CScreenState(aScheduler, aScreenOrdinal);
   286 	CleanupStack::PushL(self);
   287 	self->ConstructL();
   288 	CleanupStack::Pop(self);
   289 	return self;	
   290 	}
   291 
   292 void CWindowServer::CDefaultAnimationScheduler::CScreenState::ConstructL ()
   293 	{
   294 	iUpdateOn = CPeriodic::NewL(EComposeCompletePriority);
   295 	iWaitingClients.ReserveL(8);
   296 	CActiveScheduler::Add(this);
   297 	}
   298 
   299 CWindowServer::CDefaultAnimationScheduler::CScreenState::~CScreenState()
   300 	{
   301 	CActive::Cancel();
   302 	iInvalidated.Close();
   303 	delete iUpdateOn;
   304 	TInt i = iWaitingClients.Count();
   305 	while(i--)
   306 		{
   307 		iWaitingClients[i].iObserver.ScreenUpdateComplete(KErrAbort);
   308 		}
   309 	iWaitingClients.Close();
   310 	}
   311 
   312 void CWindowServer::CDefaultAnimationScheduler::CScreenState::SetActive()
   313 	{
   314 	CActive::SetActive ();
   315 	}
   316 
   317 /**
   318 This function is called from CWsClient d'tor to make sure we will not hang on to any deleted objects. 
   319 */
   320 void CWindowServer::CDefaultAnimationScheduler::CScreenState::ClearScreenUpdateObserver(const MWsAnimationScheduler::MScreenUpdateObserver& aObserver)
   321 	{
   322 	const TInt count = iWaitingClients.Count();
   323 	for(TInt i = count-1 ; i >= 0; i--)
   324 		{
   325 		if( &aObserver == &(iWaitingClients[i].iObserver) )
   326 			{
   327 			TWaitingClient& client = iWaitingClients[i];
   328 			client.iObserver.ScreenUpdateComplete(KErrCancel);
   329 			iWaitingClients.Remove(i);
   330 			}
   331 		}
   332 	}
   333 
   334 void CWindowServer::CDefaultAnimationScheduler::CScreenState::WaitForRedraws(MWsAnimationScheduler::MScreenUpdateObserver& aObserver, TInt aNumRedraws)
   335 	{
   336 	const TUint targetFrame = iFrameCount + aNumRedraws;
   337 	TWaitingClient request(aObserver, targetFrame);
   338 	TInt err = iWaitingClients.Append(request);
   339 	if(err != KErrNone)
   340 		{
   341 		//If OOM and already have 8 waiting clients we will not accept a 9th client
   342 		aObserver.ScreenUpdateComplete(KErrNoMemory);
   343 		}
   344 	}
   345 
   346 void CWindowServer::CDefaultAnimationScheduler::CScreenState::ReleaseRemainingClients()
   347 	{
   348 	const TInt count = iWaitingClients.Count();
   349 	for(TInt i = count-1; i >= 0; i--)
   350 		{
   351 		TWaitingClient& client = iWaitingClients[i];
   352 		client.iObserver.ScreenUpdateComplete(KErrNone);
   353 		iWaitingClients.Remove(i);
   354 		}
   355 	}
   356 
   357 void CWindowServer::CDefaultAnimationScheduler::CScreenState::ReleaseClientsWaitingFor(TUint aCurrentFrame)
   358 	{
   359 	const TInt count = iWaitingClients.Count();
   360 	for(TInt i = count-1; i >= 0; i--)
   361 		{
   362 		TWaitingClient& client = iWaitingClients[i];
   363 		if(aCurrentFrame == client.iTargetFrame)
   364 			{
   365 			client.iObserver.ScreenUpdateComplete(KErrNone);
   366 			iWaitingClients.Remove(i);
   367 			}
   368 		}
   369 	}
   370 
   371 /** 
   372 Invoked when the rendering pipline signals that it is ready to receive updates.
   373 */
   374 void CWindowServer::CDefaultAnimationScheduler::CScreenState::RunL() 
   375 	{
   376 	iFrameCount++;
   377 	
   378 	//Complete any clients waiting for this frame
   379 	ReleaseClientsWaitingFor(iFrameCount);
   380 	
   381 	iScreenUpdateDetails.iScheduler->ProcessUpdateCompletion (iScreenUpdateDetails.iScreenNumber);
   382 	
   383 	if(!IsActive())
   384 		{
   385 		//No further pending frames, release all remaining clients
   386 		ReleaseRemainingClients();
   387 		}
   388 	
   389 	}
   390 
   391 void CWindowServer::CDefaultAnimationScheduler::ProcessUpdateCompletion (TInt aScreenNumber)
   392 	{
   393 	// Stop all waitloops related to the specified screen.
   394 	TInt waitLoopCount = iRedrawWaitLoop.Count();
   395 	for (TInt waitLoop = 0; waitLoop < waitLoopCount; waitLoop++)
   396 		if (iRedrawWaitLoop[waitLoop]->IsStarted() && (iRedrawWaitLoop[waitLoop]->iScreenNumber == aScreenNumber))
   397 			iRedrawWaitLoop[waitLoop]->AsyncStop();
   398 	
   399 	// Perform any outstanding redraws on the specified screen.
   400 	ScheduleUpdate (aScreenNumber, ETrue);
   401 	}
   402 
   403 void CWindowServer::CDefaultAnimationScheduler::DoRedrawNow(MWsScreen& aScreen, MWsAnimationScheduler::MScreenUpdateObserver& aObserver)
   404 	{
   405 	TInt screenNumber = ScreenNumber (aScreen);
   406 	TInt redrawCount = 0;
   407 	
   408 	// redrawCount is the number of times we should wait for redraws to complete.
   409 	// If a redraw is not currently active then we need to wait (at most) once: for
   410 	// any outstanding scheduled update to complete.
   411 	// If a redraw is currently active then we need to wait (at most) twice: once for
   412 	// the current update to complete, and once for any outstanding scheduled update to complete.   
   413 	if (!iScreenState[screenNumber]->IsActive())
   414 		{
   415 		// No animation in progress, so force a redraw of due updates.
   416 		ScheduleUpdate(screenNumber, ETrue);
   417 
   418 		// If there is still nothing drawing, set redrawCount to zero to make sure we do not wait.
   419 		if (!iScreenState[screenNumber]->IsActive())
   420 			{
   421 			redrawCount = 0;
   422 			aObserver.ScreenUpdateComplete(KErrNone);
   423 			}
   424 		else
   425 			{
   426 			redrawCount = 1;
   427 			iScreenState[screenNumber]->WaitForRedraws(aObserver, redrawCount);
   428 			}
   429 		}
   430 	else
   431 		{
   432 		redrawCount = 2;
   433 		iScreenState[screenNumber]->WaitForRedraws(aObserver, redrawCount);
   434 		}
   435 	}
   436 
   437 void CWindowServer::CDefaultAnimationScheduler::ClearScreenUpdateObserver(const MWsAnimationScheduler::MScreenUpdateObserver& aObserver)
   438 	{
   439 	const TInt count = iScreenState.Count();
   440 	for(TInt screenNumber=0; screenNumber<count; screenNumber++) 
   441 		{
   442 		iScreenState[screenNumber]->ClearScreenUpdateObserver(aObserver);
   443 		}
   444 	}
   445 
   446 /**
   447 Switch to deactivate animation or drawing (based on setting of iInactivityBehaviour).
   448 See InvokeDueAnimation().
   449 */
   450 void CWindowServer::CDefaultAnimationScheduler::OnInactive()
   451 	{
   452 	iInactive = ETrue;
   453 	}
   454 
   455 /**
   456 Switch to deactivate/activate animation or drawing (based on setting of iInactivityBehaviour).
   457 See InvokeDueAnimation().
   458 */	
   459 void CWindowServer::CDefaultAnimationScheduler::OnActive()
   460 	{
   461 	iInactive = EFalse;
   462 	if(iInactiveDraws)
   463 		{
   464 		iInactiveDraws = EFalse;
   465 		iIdleInitiator->CallBack();
   466 		}
   467 	}
   468 
   469 void CWindowServer::CDefaultAnimationScheduler::ScheduleRedraw(MWsScreen& aScreen,const TTime& aWhen)
   470 	{
   471 	iRedrawScheduled = ETrue;
   472 	ScheduleAnimation(aScreen, aWhen);
   473 	}
   474 
   475 /** 
   476 Given a MWsScreen pointer, return an integer value representing the ordinal position
   477 of the screen in the Window Server.
   478 */
   479 TInt CWindowServer::CDefaultAnimationScheduler::ScreenNumber(MWsScreen& aScreen) const
   480 	{
   481 	TInt numberOfScreens = CWsTop::NumberOfScreens();
   482 	TInt theScreen;
   483 	
   484 	for (theScreen = 0; theScreen < numberOfScreens; theScreen++)
   485 		if (CWsTop::Screen(theScreen) == &aScreen) 
   486 			break;
   487 	
   488 	WS_ASSERT_DEBUG(theScreen < numberOfScreens, EWsPanicWsGraphic);
   489 	return theScreen;
   490 	}
   491 
   492 /**
   493 Perform redraw and return only when completed.
   494 NOTE: This method uses CActiveSchedulerWait to run a "modal wait loop" while the
   495 		redraw complete signal is pending. When the signal is received, AsyncStop() is
   496 		invoked on all active wait loops for the signalling screen. 
   497 */
   498 void CWindowServer::CDefaultAnimationScheduler::DoRedrawNow(MWsScreen& aScreen)
   499 	{	
   500 	TInt screenNumber = ScreenNumber (aScreen);
   501 	TInt redrawCount = 0;
   502 	
   503 	// redrawCount is the number of times we should wait for redraws to complete.
   504 	// If a redraw is not currently active then we need to wait (at most) once: for
   505 	// any outstanding scheduled update to complete.
   506 	// If a redraw is currently active then we need to wait (at most) twice: once for
   507 	// the current update to complete, and once for any outstanding scheduled update to complete.	
   508 	if (!iScreenState[screenNumber]->IsActive())
   509 		{		
   510 		// No animation in progress, so force a redraw of due updates.
   511 		ScheduleUpdate(screenNumber, ETrue);
   512 		
   513 		// If there is still nothing drawing, set redrawCount to zero to make sure we do not wait.
   514 		if (!iScreenState[screenNumber]->IsActive())
   515 			redrawCount = 0;
   516 		else
   517 			redrawCount = 1;
   518 		}
   519 	else
   520 		redrawCount = 2;
   521 	
   522 	// Wait for the sepecified number of redraws.
   523 	if (redrawCount)
   524 		{
   525 		// Allocate the wait loop on the stack so we are not subject to heap OOM.
   526 		TBuf8<sizeof (CScreenUpdateWait)> buf;
   527 		Mem::FillZ(&buf, sizeof (CScreenUpdateWait));
   528 		CScreenUpdateWait* waitLoop = new (&buf) CScreenUpdateWait(screenNumber);
   529 		if (iRedrawWaitLoop.Append(waitLoop) == KErrNone)
   530 			{
   531 			// Run the active scheduler while updates are active
   532 			while (redrawCount-- && iScreenState[screenNumber]->IsActive())
   533 				waitLoop->Start();
   534 			
   535 			iRedrawWaitLoop.Remove(iRedrawWaitLoop.Count() - 1);			
   536 			}
   537 		waitLoop->~CScreenUpdateWait();
   538 		}
   539 	}
   540 
   541 /**
   542 Schedule an update for a specific screen at a given point in time.
   543 */ 
   544 void CWindowServer::CDefaultAnimationScheduler::ScheduleAnimation(MWsScreen& aScreen, const TTime& aWhen)
   545 	{	
   546 	TSchedule schedule;
   547 	schedule.iScreen = &aScreen;
   548 	schedule.iScheduled = ETrue;
   549 	schedule.iWhen = aWhen;
   550 	schedule.iScreenNumber = ScreenNumber (aScreen);
   551 	schedule.iRedraw = iRedrawScheduled;
   552 	iRedrawScheduled = EFalse;
   553 	TBool ok = EFalse;
   554 	const TInt idx = iSchedule.FindInUnsignedKeyOrder(schedule);
   555 	if(0 <= idx)
   556 		{
   557 		TSchedule& currSchedule=iSchedule[idx];
   558 		if(currSchedule.iScheduled)
   559 			{
   560 			if(currSchedule.iWhen > aWhen)
   561 				{
   562 				currSchedule.iWhen = aWhen;
   563 				}
   564 			}
   565 		else
   566 			{
   567 			currSchedule = schedule;
   568 			}
   569 		ok = ETrue;
   570 		}
   571 	else
   572 		ok = (KErrNone == iSchedule.InsertInUnsignedKeyOrder(schedule));
   573 
   574 	if(ok)
   575 		iIdleInitiator->CallBack();
   576 	}
   577 
   578 void CWindowServer::CDefaultAnimationScheduler::UnscheduleAnimation(MWsScreen& aScreen)
   579 	{
   580 	TSchedule schedule;
   581 	schedule.iScreen = &aScreen;
   582 	const TInt idx = iSchedule.FindInUnsignedKeyOrder(schedule);
   583 	if(0 <= idx)
   584 		iSchedule[idx].iScheduled = EFalse;
   585 	}
   586 
   587 TBool CWindowServer::CDefaultAnimationScheduler::OnIdleCallBack(TAny* aAny)
   588 	{
   589 	WS_ASSERT_DEBUG(aAny, EWsPanicWsGraphic);
   590 	
   591 	if(aAny)
   592 		static_cast<CDefaultAnimationScheduler*>(aAny)->OnIdleCallBack(EFalse);
   593 
   594 	return EFalse; //ignored by caller
   595 	}
   596 
   597 void CWindowServer::CDefaultAnimationScheduler::OnIdleCallBack(TBool aForce)
   598 	{
   599 	const TInt screenCount = iEnv.ScreenCount();
   600 	for(TInt ii = 0; ii < screenCount; ii++)
   601 		ScheduleUpdate (ii, aForce);
   602 	}
   603 
   604 
   605 
   606 /** 
   607 @return The number of microseconds (from now) that the specified scheduled update should be run at. This
   608 takes into account any set grace period and protects the scheduler from entering an infinite loop servicing
   609 animations with back-to-back frame updates.
   610 */
   611 TTimeIntervalMicroSeconds CWindowServer::CDefaultAnimationScheduler::GetDueDelta (TBool aForceRedraw, TSchedule* aScheduledUpdate)
   612 	{
   613 	WS_ASSERT_DEBUG(aScheduledUpdate, EWsPanicWsGraphic);
   614 	WS_ASSERT_DEBUG(aScheduledUpdate->iScheduled, EWsPanicWsGraphic);
   615 	
   616 	TTime now;
   617 	TInt64 grace = I64LIT(0); 
   618 	TTimeIntervalMicroSeconds thisUpdateDueIn = I64LIT(0); //Microseconds from now
   619 	
   620 	// Only use grace periods if not forcing due updates.
   621 	if (!aForceRedraw)
   622 		{
   623 		if (aScheduledUpdate->iRedraw)
   624 			grace = iRedrawGracePeriod;
   625 		else
   626 			grace = iAnimationGracePeriod;
   627 		}
   628 	
   629 	now.UniversalTime();
   630 	thisUpdateDueIn = aScheduledUpdate->iWhen.MicroSecondsFrom(now);
   631 	
   632 	// Add the grace period if the update is due in less time than the grace period.
   633 	if (thisUpdateDueIn < grace)
   634 		thisUpdateDueIn = grace;
   635 	else if (thisUpdateDueIn > KHalfHour)
   636 		thisUpdateDueIn = KHalfHour;
   637 	
   638 	return thisUpdateDueIn;
   639 	}
   640 
   641 /**
   642 Schedule an actual screen update at the point in time at which it is due. The due time may be modified by
   643 this method based on any "grace period" values.
   644  
   645 @param aScreen		Screen number to update.
   646 @param aForceRedraw Force redraws that are due. This causes grace periods not to be used.
   647 */
   648 void CWindowServer::CDefaultAnimationScheduler::ScheduleUpdate (TInt aScreenNumber, TBool aForceRedraw)
   649 	{	
   650 	// Schedule updates for any invalidated regions.
   651 	RedrawAllInvalidatedRegions (aScreenNumber);
   652 	
   653 	TSchedule* scheduledUpdate = GetScheduledScreenUpdate(aScreenNumber);
   654 	if (scheduledUpdate)
   655 		{
   656 		WS_ASSERT_DEBUG(scheduledUpdate->iScheduled, EWsPanicWsGraphic);
   657 		WS_ASSERT_DEBUG(aScreenNumber < iScreenState.Count(), EWsPanicWsGraphic);
   658 		
   659 		CScreenState& screenState = *iScreenState[aScreenNumber];
   660 				
   661 		// Initiate redraw if scheduled and not currently updating the display.
   662 		if(!screenState.IsActive())
   663 			{
   664 			TTimeIntervalMicroSeconds thisUpdateDueIn =	
   665 				GetDueDelta (aForceRedraw, scheduledUpdate);
   666 			
   667 			// Reschedule any preexisting update if this one is due earlier.
   668 			// If this update is not due earlier than a preexisting update then 
   669 			// there is nothing to do - just let the pending update occur.
   670 			TTime now;
   671 			now.UniversalTime();
   672 			TBool performUpdate = ETrue;
   673 			if(screenState.iUpdateOn->IsActive())
   674 				{
   675 				if (thisUpdateDueIn < screenState.iExpectedTickTime.MicroSecondsFrom(now))
   676 					screenState.iUpdateOn->Cancel();
   677 				else
   678 					performUpdate = EFalse;
   679 				}
   680 			
   681 			if (performUpdate)
   682 				{
   683 				if (thisUpdateDueIn.Int64() == 0) // Perform an immediate update if we are due.
   684 					{
   685 					screenState.iExpectedTickTime = now;
   686 					InvokeDueAnimation(aScreenNumber);
   687 					}
   688 				else // Schedule the tick at the appropriate time.
   689 					{
   690 					WS_ASSERT_DEBUG(thisUpdateDueIn.Int64() > 0, EWsPanicWsGraphic);
   691 					screenState.iExpectedTickTime = now + thisUpdateDueIn;
   692 					screenState.iUpdateOn->Start(thisUpdateDueIn.Int64(),0,TCallBack(InvokeDueAnimation, &screenState.iScreenUpdateDetails));
   693 					}
   694 				}
   695 			}
   696 		}
   697 	}
   698 
   699 /**
   700 @return 	A pointer to the scheduled update details currently associated with the specified screen.
   701 			If there is no scheduled update then NULL is returned.
   702 @note 		There is only ever one scheduled update per screen.
   703 */
   704 CWindowServer::CDefaultAnimationScheduler::TSchedule* CWindowServer::CDefaultAnimationScheduler::GetScheduledScreenUpdate(TInt aScreenNumber)
   705 	{
   706 	TSchedule* result = NULL;
   707 	const TInt count = iSchedule.Count();
   708 	for(TInt ii = 0; ii < count; ii++)
   709 		{
   710 		if (iSchedule[ii].iScreenNumber == aScreenNumber)
   711 			{
   712 			if (iSchedule[ii].iScheduled)
   713 				result = &iSchedule[ii];
   714 			break;
   715 			}
   716 		}
   717 	
   718 	return result;
   719 	}
   720 
   721 /**
   722 Redraw invalidated graphic IDs. If invalid regions exist, this will cause ScheduleRedraw() to be invoked.
   723 */ 
   724 void CWindowServer::CDefaultAnimationScheduler::RedrawAllInvalidatedRegions (TInt aScreen)
   725 	{	
   726 	WS_ASSERT_DEBUG(iScreenState.Count() > aScreen, EWsPanicWsGraphic);
   727 	
   728 	CScreenState& screenState = *iScreenState[aScreen];
   729 	if(screenState.iInvalidateAll || screenState.iInvalidated.Count())
   730 		{		
   731 		const TArray<TGraphicDrawerId> invalidArray = screenState.iInvalidated.Array();
   732 		MWsScreen* screen = iEnv.Screen(aScreen);
   733 		WS_ASSERT_DEBUG(screen, EWsPanicWsGraphic);
   734 		if(screen)
   735 			{
   736 			if(screenState.iInvalidateAll)
   737 				Redraw(*screen);
   738 			else
   739 				RedrawInvalid(*screen, screenState.iInvalidated.Array());
   740 			}
   741 		screenState.iInvalidateAll = EFalse;
   742 		}
   743 	screenState.iInvalidated.Reset();
   744 	}
   745 	
   746 TBool CWindowServer::CDefaultAnimationScheduler::InvokeDueAnimation(TAny* aAny)
   747 	{
   748 	WS_ASSERT_DEBUG(aAny, EWsPanicWsGraphic);
   749 	TScreenUpdateDetails* args = reinterpret_cast<TScreenUpdateDetails*>(aAny);
   750 	if(args)
   751 		args->iScheduler->InvokeDueAnimation (args->iScreenNumber);
   752 
   753 	return EFalse;
   754 	}
   755 
   756 void CWindowServer::CDefaultAnimationScheduler::InvokeDueAnimation(TInt aScreen)
   757 	{
   758 	WS_ASSERT_DEBUG(aScreen < iScreenState.Count(), EWsPanicWsGraphic);
   759 	CScreenState& screenState = *iScreenState[aScreen];
   760 	WS_ASSERT_DEBUG(!screenState.IsActive(), EWsPanicWsGraphic);
   761 
   762 	// All updates are driven through ScheduleRedraw() and ScheduleAnimation().
   763 	screenState.iUpdateOn->Cancel();
   764 	
   765 	TSchedule* scheduledUpdate = GetScheduledScreenUpdate(aScreen);	
   766 	if (scheduledUpdate)
   767 		{
   768 		WS_ASSERT_DEBUG(scheduledUpdate->iScheduled, EWsPanicWsGraphic);
   769 		
   770 		// Honour any flags that indicate we should not redraw. 
   771 		switch(iInactivityBehaviour)
   772 			{
   773 			case EStopAnimation :
   774 				// Stop server side drawing. Only the client may redraw if iInactive is set. 
   775 				if(iInactive && !scheduledUpdate->iRedraw)
   776 					{
   777 					iInactiveDraws = ETrue;
   778 					return;
   779 					}
   780 				break;
   781 			case EStopAllDrawing :
   782 				// Stop both client and server side drawing.
   783 				if(iInactive) 
   784 					{
   785 					iInactiveDraws = ETrue;
   786 					return;
   787 					}
   788 				break;
   789 			case EIgnore :
   790 				break;
   791 			default :
   792 				WS_ASSERT_DEBUG(EFalse, EWsPanicWsGraphic);
   793 				break;
   794 			}
   795 	
   796 		scheduledUpdate->iScheduled = EFalse;
   797 		screenState.SetActive();
   798 		Animate(*scheduledUpdate->iScreen, &(screenState.iStatus));
   799 		}
   800 	}
   801 
   802 // CWindowServer::CServer \\\\\\\\\\\\\\\\\\\\\\
   803 
   804 class CWindowServer::CServer : public CPolicyServer
   805 	{
   806 public:
   807 	static CServer* NewL()
   808 		{
   809 		return new(ELeave) CServer;
   810 		}
   811 	void StartL()
   812 		{
   813 		CPolicyServer::StartL(KWSERVServerName);
   814 		}
   815 	TInt SessionCount()
   816 		{
   817 		iSessionIter.SetToFirst();
   818 		TInt count=0;
   819 		while(iSessionIter++)
   820 			++count;
   821 		return(count);
   822 		}
   823 
   824 public: //from CPolicyServer
   825 	/** Creates a new client for this server. */
   826 	CSession2* NewSessionL(const TVersion& aVersion, const RMessage2& aMessage) const
   827 		{
   828 		TVersion v(KWservMajorVersionNumber, KWservMinorVersionNumber, KWservBuildVersionNumber);
   829 		if (User::QueryVersionSupported(v, aVersion)==EFalse)
   830 			User::Leave(KErrNotSupported);
   831 		RThread thread;
   832 		User::LeaveIfError(aMessage.Client(thread));
   833 		return(new(ELeave) CWsClient(thread));
   834 		}
   835 private:
   836 	CServer() : CPolicyServer(EMainServerPriority, KWsServPolicy)
   837 	{}
   838 	};
   839 
   840 
   841 // CWindowServer \\\\\\\\\\\\\\\\\\\\\\\\\\\
   842 
   843 CWindowServer *CWindowServer::NewL()
   844 //
   845 // Create a new CWindowServer.
   846 //
   847 	{
   848 	CWindowServer* self = new(ELeave) CWindowServer();
   849 	CleanupStack::PushL(self);
   850 	self->ConstructL();
   851 	CleanupStack::Pop(self);
   852 	return self;
   853 	}
   854 
   855 CWindowServer::CWindowServer()
   856 //
   857 // Constructor.
   858 //
   859 	{
   860 	}
   861 
   862 CWindowServer::~CWindowServer()
   863 	{
   864 	delete iServer;
   865 
   866 	iMemoryReleases.Reset();
   867 	WS_ASSERT_DEBUG(iDrawerMasterIndex.IsEmpty(), EWsPanicWsGraphic);
   868 	iDrawerMasterIndex.Close();
   869 	
   870 	delete iDefaultAnimationScheduler;
   871 	iDefaultAnimationScheduler = NULL; //might be called from clients during server destruction	
   872 	}
   873 	
   874 void CWindowServer::ConstructL()
   875 	{
   876 	iServer = CServer::NewL();
   877 	CWsTop::PluginManager()->InitializePluginsL(*this); // plugins are loaded and own by CWsTop
   878 	iDefaultAnimationScheduler = new(ELeave) CDefaultAnimationScheduler(*this);
   879 	iDefaultAnimationScheduler->ConstructL();
   880 	RegisterMemoryRelease(this);
   881 	}
   882 
   883 void CWindowServer::StartL()
   884 	{
   885 	iServer->StartL();
   886 	}
   887 
   888 void CWindowServer::SetPinClientDescriptors(TBool aPin)
   889 	{
   890 	iServer->SetPinClientDescriptors(aPin);
   891 	}
   892 
   893 TInt CWindowServer::SessionCount()
   894 	{
   895 	return iServer->SessionCount();
   896 	}
   897 
   898 const CWsGraphicDrawer* CWindowServer::ResolveGraphic(const TGraphicDrawerId& aId) const
   899 	{
   900 	return iDrawerMasterIndex.ResolveGraphic(aId);
   901 	}
   902 	
   903 void CWindowServer::Invalidate(const TGraphicDrawerId& aId)
   904 	{
   905 	AnimationScheduler()->Invalidate(aId);
   906 	}
   907 	
   908 TInt CWindowServer::ScreenCount() const
   909 	{
   910 	return CWsTop::NumberOfScreens();
   911 	}
   912 
   913 MWsScreen* CWindowServer::Screen(TInt aIndex)
   914 	{
   915 	if((aIndex >= 0) && (aIndex < ScreenCount()))
   916 		{
   917 		return CWsTop::Screen(aIndex);
   918 		}
   919 	return NULL;
   920 	}
   921 	
   922 const MWsScreen* CWindowServer::Screen(TInt aIndex) const
   923 	{
   924 	if((aIndex >= 0) && (aIndex < ScreenCount()))
   925 		{
   926 		return CWsTop::Screen(aIndex);
   927 		}
   928 	return NULL;
   929 	}
   930 	
   931 /**
   932 Custom Animation Scheduler
   933 */
   934 TBool CWindowServer::SetCustomAnimationScheduler(MWsAnimationScheduler* /*aScheduler*/)
   935 	{
   936 	return EFalse;
   937 	}
   938 
   939 TBool CWindowServer::HasCustomAnimationScheduler() const
   940 	{
   941 	return EFalse;
   942 	}
   943 
   944 TBool CWindowServer::ClearCustomAnimationScheduler(MWsAnimationScheduler* /*aCurrentScheduler*/)
   945 	{
   946 	return EFalse;
   947 	}
   948 	
   949 MWsAnimationScheduler* CWindowServer::AnimationScheduler()
   950 	{
   951 	return iDefaultAnimationScheduler;
   952 	}
   953 
   954 void CWindowServer::PrepareShutdown()
   955 	{
   956 	//Stop the renderloop, i.e. prevent any further calls to MWsAnimationScheduler::Animate() 
   957 	delete iDefaultAnimationScheduler;
   958 	iDefaultAnimationScheduler = NULL;
   959 	}
   960 
   961 TInt CWindowServer::RegisterEventHandler(CWsGraphicDrawer* aDrawer, MWsEventHandler* aHandler, TUint32 aEventMask)
   962 	{
   963 	if (!aDrawer || !aHandler || aEventMask==0)
   964 		return KErrArgument;
   965 	TInt err = TWindowServerEvent::RegisterDrawerHandler(aDrawer, aEventMask);
   966 	if (err != KErrNone)
   967 		return err;
   968 	aDrawer->SetEventHandler(aHandler);
   969 	return KErrNone;
   970 	}
   971 	
   972 TInt CWindowServer::UnregisterEventHandler(CWsGraphicDrawer* aDrawer)
   973 	{
   974 	if (!aDrawer || (aDrawer && !aDrawer->HasEventHandler()))
   975 		return KErrArgument;
   976 	TInt err = TWindowServerEvent::UnregisterDrawerHandler(aDrawer);
   977 	if (err != KErrNone)
   978 		return err;
   979 	aDrawer->SetEventHandler(NULL);
   980 	return KErrNone;
   981 	}
   982 	
   983 TInt CWindowServer::RegisterWsEventHandler(MWsEventHandler* aHandler, TUint32 aEventMask)
   984 	{
   985 	if (!aHandler || aEventMask==0)
   986 		return KErrArgument;
   987 	return TWindowServerEvent::RegisterWsEventHandler(aHandler, aEventMask);
   988 	}
   989 	
   990 TInt CWindowServer::UnregisterWsEventHandler(MWsEventHandler* aHandler)
   991 	{
   992 	return TWindowServerEvent::UnregisterWsEventHandler(aHandler);
   993 	}
   994 	
   995 TAny* CWindowServer::ResolveObjectInterface(TUint aTypeId)
   996 	{
   997 	switch(aTypeId)
   998 		{
   999 		case MWsActiveSchedulerDebug::EWsObjectInterfaceId:
  1000 			return static_cast<MWsActiveSchedulerDebug*>(CWsActiveScheduler::Static());
  1001 		case MWsIniFile::EWsObjectInterfaceId:
  1002 			return static_cast<MWsIniFile*>(WsIniFile);
  1003 		}
  1004 	
  1005 	if (CWsPluginManager *plugMgr=CWsTop::PluginManager())
  1006 		return plugMgr->ResolveObjectInterface(aTypeId);
  1007 	
  1008 	return NULL;
  1009 	}
  1010 
  1011 void CWindowServer::Log(TInt aPriority,const TDesC &aFmt,TInt aParam)
  1012 	{
  1013 	if (wsDebugLog)
  1014 		{
  1015 		wsDebugLog->MiscMessage(aPriority, aFmt, aParam);
  1016 		}
  1017 	}
  1018 
  1019 // CWsGraphicDrawer master index
  1020 
  1021 TInt CWindowServer::AddGraphicDrawer(CWsGraphicDrawer* aDrawer)
  1022 	{
  1023 	return iDrawerMasterIndex.Add(aDrawer);
  1024 	}
  1025 
  1026 TInt CWindowServer::SwapGraphicDrawer(CWsGraphicDrawer* aDrawer)
  1027 	{
  1028 	return iDrawerMasterIndex.Swap(aDrawer);
  1029 	}
  1030 	
  1031 TInt CWindowServer::RemoveGraphicDrawer(const TGraphicDrawerId& aId)
  1032 	{
  1033 	return iDrawerMasterIndex.Remove(aId);
  1034 	}
  1035 
  1036 TInt CWindowServer::RemoveAllGraphicDrawers(const MWsClient& aOwner)
  1037 	{
  1038 	return iDrawerMasterIndex.RemoveAll(aOwner);
  1039 	}
  1040 
  1041 TInt CWindowServer::RegisterMemoryRelease(MWsMemoryRelease * aMemoryRelease)
  1042 	{
  1043 	return iMemoryReleases.Append(aMemoryRelease);
  1044 	}
  1045 
  1046 void CWindowServer::UnregisterMemoryRelease(MWsMemoryRelease * aMemoryRelease)
  1047 	{
  1048 	for (TInt ii = iMemoryReleases.Count() - 1; ii >= 0; --ii)
  1049 		{
  1050 		if (iMemoryReleases[ii] == aMemoryRelease)
  1051 			{
  1052 			iMemoryReleases.Remove(ii);
  1053 			break;
  1054 			}
  1055 		}
  1056 	}
  1057 
  1058 TBool CWindowServer::ReleaseMemory(TMemoryReleaseLevel aLevel)
  1059 	{
  1060 	return CWsWindow::ReleaseMemory(aLevel);
  1061 	}
  1062 
  1063 TBool CWindowServer::ReleaseMemory()
  1064 	{
  1065 	TBool released = EFalse;
  1066 	for (TInt level = MWsMemoryRelease::ELow; !released && level <= MWsMemoryRelease::EHigh; ++level)
  1067 		{
  1068 		for (TInt ii = iMemoryReleases.Count() - 1; !released && ii >= 0; --ii)
  1069 			{
  1070 			released = iMemoryReleases[ii]->ReleaseMemory(static_cast<TMemoryReleaseLevel>(level));
  1071 			}
  1072 		}
  1073 	return released;
  1074 	}
  1075 
  1076 void CWindowServer::DestroySessionsForShutdown()
  1077 	{
  1078 	delete iServer;
  1079 	iServer = NULL;
  1080 	}
  1081