os/graphics/windowing/windowserver/nonnga/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 
    27 GLREF_D CDebugLogBase *wsDebugLog;
    28 
    29 const TUint KRangeCount = 1; 
    30 // We use a lot of 64 bit time calculations, but a periodic can only cope with signed 32 bit times
    31 // which gives them a limit of about 35 minutes.
    32 // Fortunately, our animtions are safe if redrawn early.  Every half an hour isn't going to hurt.
    33 const TTimeIntervalMicroSeconds KHalfHour = 30 * 60 * 1000 * 1000;
    34 
    35 const TInt KWsServRanges[KRangeCount] = 
    36 	{	
    37 	0
    38 	};
    39 
    40 const TUint8 KElementsIndex[KRangeCount] =
    41 	{
    42 	CPolicyServer::EAlwaysPass,		
    43 	};
    44 
    45 const CPolicyServer::TPolicyElement KPolicyElements[] = 
    46 	{ 
    47 	{_INIT_SECURITY_POLICY_C1(ECapabilityPowerMgmt), CPolicyServer::EFailClient}, 
    48 	{_INIT_SECURITY_POLICY_C1(ECapabilitySwEvent), CPolicyServer::EFailClient}, 
    49 	{_INIT_SECURITY_POLICY_C1(ECapabilityWriteDeviceData), CPolicyServer::EFailClient} 
    50 	};
    51 
    52 const CPolicyServer::TPolicy KWsServPolicy =
    53 	{
    54 	CPolicyServer::EAlwaysPass, 
    55 	KRangeCount,
    56 	KWsServRanges,
    57 	KElementsIndex,
    58 	KPolicyElements 	
    59 	};
    60  	
    61 // CWindowServer::CDefaultAnimationScheduler \\\\\\\\\\\\\\\\\\\\\\\\\\\
    62 
    63 class CWindowServer::CDefaultAnimationScheduler: public CBase, public MWsAnimationScheduler
    64 	{
    65 public:
    66 	enum TInactivityBehaviour
    67 		{
    68 		EStopAnimation,
    69 		EStopAllDrawing,
    70 		EIgnore,
    71 		};
    72 	class CKickBack;
    73 	CDefaultAnimationScheduler(MWsGraphicDrawerEnvironment& aEnv);
    74 	~CDefaultAnimationScheduler();
    75 	void ConstructL();		//LeaveScan:  member of nested class declaration
    76 	// implementing MWsAnimationScheduler
    77 	void ScheduleAnimation(MWsScreen& aScreen,const TTime& aWhen);
    78 	void UnscheduleAnimation(MWsScreen& aScreen);
    79 	void Invalidate(const TGraphicDrawerId& aId);
    80 	void OnInactive();
    81 	void OnActive();
    82 	void ScheduleRedraw(MWsScreen& aScreen,const TTime& aWhen);
    83 	void DoRedrawNow(MWsScreen& aScreen);
    84 	void DoRedrawNow(MWsScreen& aScreen, MWsAnimationScheduler::MScreenUpdateObserver& aObserver);
    85 	void ClearScreenUpdateObserver(const MWsAnimationScheduler::MScreenUpdateObserver& /*aObserver*/) {}
    86 	TInt RemoveGraphicDrawer(const TGraphicDrawerId &aId);
    87 	
    88 private:
    89 	static TBool OnIdleCallBack(TAny* aAny);
    90 	void OnIdleCallBack(TBool aForceRedraw);
    91 	static TBool OnTickCallBack(TAny* aAny);
    92 	void OnTickCallBack();
    93 	static TBool OnKickBack(TAny* aAny);
    94 	void OnKickBack();
    95 private:
    96 	MWsGraphicDrawerEnvironment& iEnv;
    97 	static const TInt64 KRedrawGrace;
    98 	static const TInt64 KAnimationGrace;
    99 	static const TInt64 KDsaAnimationGrace;
   100 	CAsyncCallBack* iIdleInitiator;
   101 	CPeriodic* iTick;
   102 	CKickBack* iKickBack;
   103 	struct TSchedule
   104 		{
   105 		MWsScreen* iScreen; // used as a unique index, searching with FindInUnsignedKeyOrder
   106 		TBool iSchedule;
   107 		TTime iWhen;
   108 		TInt iGeneration;
   109 		};
   110 	RArray<TSchedule> iSchedule;
   111 	TInt iGeneration;
   112 	RArray<TGraphicDrawerId> iInvalidated;
   113 	TBool iInvalidateAll; // if we could not add to iInvalidated, we have to instead redraw everything
   114 	TTime iWhenDesired;
   115 	TBool iInactive;
   116 	TBool iInactiveDraws;
   117 	TTime iExpectedTime;
   118 	TBool iRedrawScheduled;
   119 	TInt64 iRedrawGracePeriod;
   120 	TInt64 iAnimationGracePeriod;
   121 	TInt64 iDsaAnimationGracePeriod;
   122 	TInactivityBehaviour iInactivityBehaviour;
   123 	};
   124 	
   125 TInt CWindowServer::CDefaultAnimationScheduler::RemoveGraphicDrawer(const TGraphicDrawerId &aId)
   126 	{
   127 	TInt index=iInvalidated.Find(aId);
   128 	if (index!=KErrNotFound)
   129 		iInvalidated.Remove(index);
   130 	return index;
   131 	}	
   132 	
   133 class CWindowServer::CDefaultAnimationScheduler::CKickBack: public CActive
   134 	{
   135 public:
   136 	CKickBack(const TCallBack& aCallBack);
   137 	void ConstructL();
   138 	void RequestKickBack();
   139 	~CKickBack();
   140 private:
   141 	static TInt IdleThreadFunc(TAny* aAny);
   142 	void Loop();
   143 	// from CActive
   144 	void RunL(); // fires when kicked back by the idle thread
   145 	void DoCancel();
   146 private:
   147 	RThread iWservThread;
   148 	RThread iIdleThread;
   149 	TRequestStatus iIdleStatus;
   150 	TCallBack iCallBack;
   151 	};
   152 	
   153 CWindowServer::CDefaultAnimationScheduler::CKickBack::CKickBack(const TCallBack& aCallBack) :
   154 CActive(EPriorityNormal),
   155 iCallBack(aCallBack)
   156 	{
   157 	CActiveScheduler::Add(this);
   158 	}
   159 	
   160 void CWindowServer::CDefaultAnimationScheduler::CKickBack::ConstructL()
   161 	{
   162 	_LIT(KIdleThreadName,"NearlyIdleKickBack");
   163 	const TInt KStackSize = 1024;
   164 	User::LeaveIfError(iWservThread.Open(iWservThread.Id()));
   165 	User::LeaveIfError(iIdleThread.Create(KIdleThreadName(),IdleThreadFunc,KStackSize,NULL,this));
   166 	iIdleThread.SetPriority(EPriorityAbsoluteVeryLow);
   167 	iIdleThread.Resume();
   168 	}
   169 	
   170 void CWindowServer::CDefaultAnimationScheduler::CKickBack::RequestKickBack()
   171 	{
   172 	if (!IsActive())
   173 		{
   174 		iStatus = KRequestPending;
   175 		SetActive();
   176 		TRequestStatus * status = &iIdleStatus;
   177 		iIdleThread.RequestComplete(status, KErrNone);
   178 		}
   179 	}
   180 
   181 void CWindowServer::CDefaultAnimationScheduler::CKickBack::Loop()
   182 	{
   183 	FOREVER
   184 		{
   185 		// This is used here for performance reasons. 
   186 		User::WaitForRequest(iIdleStatus); 
   187 		iIdleStatus = KRequestPending;
   188 		if (IsActive()&& (iStatus == KRequestPending))
   189 			{
   190 			TRequestStatus * status = &iStatus;
   191 			iWservThread.RequestComplete(status,KErrNone);
   192 			}
   193 		}
   194 	}
   195 
   196 void CWindowServer::CDefaultAnimationScheduler::CKickBack::RunL()
   197 	{
   198 	iCallBack.CallBack();
   199 	}
   200 
   201 void CWindowServer::CDefaultAnimationScheduler::CKickBack::DoCancel()
   202 	{
   203 	}
   204 
   205 CWindowServer::CDefaultAnimationScheduler::CKickBack::~CKickBack()
   206 	{
   207 	Cancel();
   208 	iWservThread.Close();
   209 	iIdleThread.Kill(0);
   210 	iIdleThread.Close();
   211 	}
   212 	
   213 TInt CWindowServer::CDefaultAnimationScheduler::CKickBack::IdleThreadFunc(TAny* aAny)
   214 	{
   215 	CKickBack* owner = reinterpret_cast<CKickBack*>(aAny);
   216 	ASSERT(owner);
   217 	if(owner)
   218 		{
   219 		owner->Loop();
   220 		}
   221 	return KErrNone;
   222 	}
   223 
   224 // If using the default animation scheduler on a device, these two numbers may be worth tweaking in the inifile
   225 // However, both are maximum periods - wserv will go faster than either if nothing else is using the system.
   226 const TInt64 CWindowServer::CDefaultAnimationScheduler::KRedrawGrace = 0; // do redraws immediately
   227 const TInt64 CWindowServer::CDefaultAnimationScheduler::KAnimationGrace = 35000; // insist upon 35ms grace for other threads to run when animating
   228 const TInt64 CWindowServer::CDefaultAnimationScheduler::KDsaAnimationGrace = 35000; // insist upon 35ms grace for other threads to run when animating during DSA
   229 	
   230 CWindowServer::CDefaultAnimationScheduler::CDefaultAnimationScheduler(MWsGraphicDrawerEnvironment& aEnv):
   231 	iEnv(aEnv), iSchedule(1,_FOFF(TSchedule,iScreen))
   232 	{
   233 	}
   234 	
   235 CWindowServer::CDefaultAnimationScheduler::~CDefaultAnimationScheduler()
   236 	{
   237 	iSchedule.Close();
   238 	iInvalidated.Close();
   239 	delete iKickBack;
   240 	delete iIdleInitiator;
   241 	delete iTick;
   242 	}
   243 	
   244 void CWindowServer::CDefaultAnimationScheduler::ConstructL()
   245 	{
   246 	_LIT(KOnInactive,"ONINACTIVE");
   247 	_LIT(KStopAnimation,"STOPANIMATION");
   248 	_LIT(KStopAllDrawing,"STOPALLDRAWING");
   249 	_LIT(KIgnore,"IGNORE");
   250 	
   251 	TPtrC inactivityBehaviourString;
   252 	WsIniFile->FindVar(KOnInactive,inactivityBehaviourString);
   253 	if(inactivityBehaviourString.CompareF(KStopAnimation)==0)
   254 		iInactivityBehaviour = EStopAnimation;
   255 	else if(inactivityBehaviourString.CompareF(KStopAllDrawing)==0)
   256 		iInactivityBehaviour = EStopAllDrawing;
   257 	else if(inactivityBehaviourString.CompareF(KIgnore)==0)
   258 		iInactivityBehaviour = EIgnore;
   259 		
   260 	_LIT(KRedrawGracePeriod, "REDRAWGRACEPERIOD");
   261 	TInt tmp = KRedrawGrace;
   262 	WsIniFile->FindVar(KRedrawGracePeriod, tmp);
   263 	iRedrawGracePeriod = tmp;
   264 	
   265 	_LIT(KAnimationGracePeriod, "ANIMATIONGRACEPERIOD");
   266 	tmp = KAnimationGrace;
   267 	WsIniFile->FindVar(KAnimationGracePeriod, tmp);
   268 	iAnimationGracePeriod = tmp;
   269 	
   270 	_LIT(KDsaAnimationGracePeriod, "DSAANIMATIONGRACEPERIOD");
   271 	tmp = KDsaAnimationGrace;
   272 	WsIniFile->FindVar(KDsaAnimationGracePeriod, tmp);
   273 	iDsaAnimationGracePeriod = tmp;
   274 	
   275 	iIdleInitiator = new(ELeave) CAsyncCallBack(TCallBack(OnIdleCallBack,this),EWsGraphicAnimateAwaitIdlePriority);
   276 	iTick = CPeriodic::NewL(EWsGraphicAnimatePriority);
   277 	
   278 	_LIT(KDisableIdleAnimation, "DISABLEIDLEANIMATION");
   279 	if (!WsIniFile->FindVar(KDisableIdleAnimation))
   280 		{
   281 		iKickBack = new CKickBack(TCallBack(OnKickBack,this));
   282 		iKickBack->ConstructL();
   283 		}
   284 	}
   285 	
   286 void CWindowServer::CDefaultAnimationScheduler::Invalidate(const TGraphicDrawerId& aId)
   287 	{
   288 	if(!iInvalidateAll)
   289 		{
   290 		switch(iInvalidated.InsertInOrder(aId,TLinearOrder<TGraphicDrawerId>(TGraphicDrawerId::Compare)))
   291 			{
   292 			case KErrNone:
   293 				break;
   294 			case KErrAlreadyExists:
   295 				break;
   296 			default:
   297 				iInvalidateAll = ETrue;
   298 				iInvalidated.Reset();
   299 			}
   300 		}
   301 	iIdleInitiator->CallBack();
   302 	}
   303 
   304 void CWindowServer::CDefaultAnimationScheduler::OnInactive()
   305 	{
   306 	iInactive = ETrue;
   307 	}
   308 	
   309 void CWindowServer::CDefaultAnimationScheduler::OnActive()
   310 	{
   311 	iInactive = EFalse;
   312 	if(iInactiveDraws)
   313 		{
   314 		iInactiveDraws = EFalse;
   315 		iIdleInitiator->CallBack();
   316 		}
   317 	}
   318 
   319 void CWindowServer::CDefaultAnimationScheduler::ScheduleRedraw(MWsScreen& aScreen,const TTime& aWhen)
   320 	{
   321 	iRedrawScheduled = ETrue;
   322 	ScheduleAnimation(aScreen, aWhen);
   323 	}
   324 	
   325 void CWindowServer::CDefaultAnimationScheduler::DoRedrawNow(MWsScreen& /*aScreen*/)
   326 	{
   327 	OnIdleCallBack(ETrue);
   328 	}
   329 	
   330 void CWindowServer::CDefaultAnimationScheduler::DoRedrawNow(MWsScreen& /*aScreen*/, MWsAnimationScheduler::MScreenUpdateObserver& aObserver)
   331 	{
   332 	OnIdleCallBack(ETrue);
   333 	aObserver.ScreenUpdateComplete(KErrNone);
   334 	}	
   335 	
   336 void CWindowServer::CDefaultAnimationScheduler::ScheduleAnimation(MWsScreen& aScreen,const TTime& aWhen)
   337 	{
   338 	TSchedule schedule;
   339 	schedule.iScreen = &aScreen;
   340 	schedule.iSchedule = ETrue;
   341 	schedule.iWhen = aWhen;
   342 	schedule.iGeneration = iGeneration;
   343 	TBool ok = EFalse;
   344 	const TInt idx = iSchedule.FindInUnsignedKeyOrder(schedule);
   345 	if(0 <= idx)
   346 		{
   347 		if(iSchedule[idx].iSchedule)
   348 			{
   349 			if(iSchedule[idx].iWhen > aWhen)
   350 				{
   351 				iSchedule[idx].iWhen = aWhen;
   352 				}
   353 			}
   354 		else
   355 			{
   356 			iSchedule[idx] = schedule;
   357 			}
   358 		iSchedule[idx].iGeneration = iGeneration;
   359 		ok = ETrue;
   360 		}
   361 	else
   362 		{
   363 		ok = (KErrNone == iSchedule.InsertInUnsignedKeyOrder(schedule));
   364 		}
   365 	if(ok)
   366 		{
   367 		//If the animation runs at very high rate which exceeds the rate WSERV can 
   368 		//perform the rendering, it is possible that WSERV animation loop will monopolize  
   369 		//processor time. 
   370   		User::After(0);	// to yeild from the animation loop 
   371 
   372 		iIdleInitiator->CallBack();
   373 		}
   374 	}
   375 
   376 void CWindowServer::CDefaultAnimationScheduler::UnscheduleAnimation(MWsScreen& aScreen)
   377 	{
   378 	TSchedule schedule;
   379 	schedule.iScreen = &aScreen;
   380 	const TInt idx = iSchedule.FindInUnsignedKeyOrder(schedule);
   381 	if(0 <= idx)
   382 		{
   383 		iSchedule[idx].iSchedule = EFalse;
   384 		}
   385 	}
   386 
   387 TBool CWindowServer::CDefaultAnimationScheduler::OnIdleCallBack(TAny* aAny)
   388 	{
   389 	CDefaultAnimationScheduler* self = reinterpret_cast<CDefaultAnimationScheduler*>(aAny);
   390 	ASSERT(self);
   391 	if(self)
   392 		{
   393 		self->OnIdleCallBack(EFalse);
   394 		}
   395 	return EFalse; //ignored by caller
   396 	}
   397 	
   398 void CWindowServer::CDefaultAnimationScheduler::OnIdleCallBack(TBool aForceRedraw)
   399 	{
   400 	TBool wasActive = EFalse;
   401 	if (iTick->IsActive())
   402 		{
   403 		wasActive = ETrue;
   404 		iTick->Cancel(); // stop ticker, as we'll reschedule if necessary
   405 		}
   406 	if (aForceRedraw)
   407 		{
   408 		// Don't need to update iExpectedTime as we will not schedule a tick.
   409 		// Don't need to update iWhenDesired as we will not schedule a kick-back.
   410 		OnTickCallBack();
   411 		return;
   412 		}
   413 	TBool tick = (iInvalidateAll || iInvalidated.Count());
   414 	TTimeIntervalMicroSeconds next = 0LL;
   415 	const TInt count = iSchedule.Count();
   416 	TTime now;
   417 	now.UniversalTime();
   418 	if(count)
   419 		{
   420 		// work out the next wanted tick
   421 		TBool animTick = EFalse;
   422 		for(TInt i=0; i<count; i++)
   423 			{
   424 			if(iSchedule[i].iSchedule)
   425 				{
   426 				const TTime when = iSchedule[i].iWhen;
   427 				const TTimeIntervalMicroSeconds fromNow = when.MicroSecondsFrom(now);
   428 				if(!animTick || (fromNow < next))
   429 					{
   430 					animTick = ETrue;
   431 					tick = ETrue;
   432 					next = fromNow;
   433 					iWhenDesired = when;
   434 					}
   435 				}
   436 			}
   437 		}
   438 	if(tick)
   439 		{
   440 		TInt64 grace;
   441 		if (iRedrawScheduled)
   442 			grace = iRedrawGracePeriod;
   443 		else if (EFalse) // DSA active
   444 			grace = iDsaAnimationGracePeriod;
   445 		else
   446 			grace = iAnimationGracePeriod;
   447 		if (wasActive)
   448 			{
   449 			TInt64 minimum = iExpectedTime.MicroSecondsFrom(now).Int64();
   450 			if (minimum >= 0 && grace > minimum)
   451 				grace = minimum;
   452 			}
   453 
   454 		if (next.Int64() <= 0)
   455 			next = 0LL ;  // No kickback/tick is needed. Make sure next == default grace period.
   456 
   457 		if(next < grace)
   458 			{
   459 			next = grace;
   460 			if (iKickBack)
   461 				iKickBack->RequestKickBack();
   462 			}
   463 		else if (next > KHalfHour)
   464 			next = KHalfHour;
   465 		
   466 		iExpectedTime=now + next;
   467 		if (next.Int64() > 0)
   468 			{
   469 			iTick->Start(next.Int64(),0,TCallBack(OnTickCallBack,this));
   470 			}
   471 		else
   472 			{
   473 			OnTickCallBack(); // scheduling for 0 doesn't actually execute immediately
   474 			}
   475 		}
   476 	}
   477 	
   478 TBool CWindowServer::CDefaultAnimationScheduler::OnKickBack(TAny* aAny)
   479 	{
   480 	CDefaultAnimationScheduler* self = reinterpret_cast<CDefaultAnimationScheduler*>(aAny);
   481 	ASSERT(self);
   482 	if(self)
   483 		{
   484 		self->OnKickBack();
   485 		}
   486 	return EFalse; //ignored by caller
   487 	}
   488 	
   489 void CWindowServer::CDefaultAnimationScheduler::OnKickBack()
   490 	{
   491 	if (iTick->IsActive())
   492 		{
   493 		iTick->Cancel();
   494 		TTime now;
   495 		now.UniversalTime();
   496 				
   497 		TTimeIntervalMicroSeconds fromNow = iWhenDesired.MicroSecondsFrom(now);
   498 
   499 		if (fromNow < 0)
   500 			{
   501 			OnTickCallBack();
   502 			}
   503 		else
   504 			{
   505 			if (fromNow > KHalfHour)
   506 				fromNow = KHalfHour;
   507 			iTick->Start(fromNow.Int64(),0,TCallBack(OnTickCallBack, this));
   508 			}
   509 		}
   510 	}
   511 	
   512 TBool CWindowServer::CDefaultAnimationScheduler::OnTickCallBack(TAny* aAny)
   513 	{
   514 	CDefaultAnimationScheduler* self = reinterpret_cast<CDefaultAnimationScheduler*>(aAny);
   515 	ASSERT(self);
   516 	if(self)
   517 		{
   518 		self->OnTickCallBack();
   519 		}
   520 	return EFalse;
   521 	}
   522 	
   523 void CWindowServer::CDefaultAnimationScheduler::OnTickCallBack()
   524 	{
   525 	iTick->Cancel();
   526 
   527 	switch(iInactivityBehaviour)
   528 		{
   529 		case EStopAnimation :
   530 			// only client redraws if inactive. server side drawing stopped.
   531 			if(iInactive && !iRedrawScheduled) 
   532 				{
   533 				iInactiveDraws = ETrue;
   534 				return;
   535 				}
   536 			break;
   537 		case EStopAllDrawing :
   538 			// if screen off, stop both client and server side drawing.
   539 			if(iInactive) 
   540 				{
   541 				iInactiveDraws = ETrue;
   542 				return;
   543 				}
   544 			break;
   545 		case EIgnore :
   546 		default :
   547 			// ignore inactivity and draw as normal
   548 			break;
   549 		}
   550 		
   551 	iRedrawScheduled = EFalse;
   552 
   553 	TTime now;
   554 	now.UniversalTime();
   555 	CWsActiveScheduler::Static()->AccumReclaimedIdleTime(Max(0LL,now.MicroSecondsFrom(iExpectedTime).Int64()));
   556 		
   557 	TInt drewCount = 0;
   558 	TInt scheduledCount = 0;
   559 	/* first redraw any screens that are affected by invalidated graphic IDs */
   560 	if(iInvalidateAll || iInvalidated.Count())
   561 		{
   562 		// cancel idle callback if it's already scheduled as we're going to redraw all invalidated 
   563 		// request at this point and clear invalidated array
   564 		if (iIdleInitiator->IsActive())
   565 			iIdleInitiator->Cancel();
   566 		const TArray<TGraphicDrawerId> invalidArray = iInvalidated.Array();
   567 		const TInt screenCount = iEnv.ScreenCount();
   568 		for(TInt i=0; i<screenCount; i++)
   569 			{
   570 			MWsScreen* screen = iEnv.Screen(i);
   571 			if(screen)
   572 				{
   573 				drewCount++;
   574 				if(iInvalidateAll)
   575 					{
   576 					Redraw(*screen);
   577 					}
   578 				else
   579 					{
   580 					RedrawInvalid(*screen,invalidArray);
   581 					}
   582 				}
   583 			}
   584 		iInvalidateAll = EFalse;
   585 		iInvalidated.Reset();
   586 		}
   587 	/* iSchedule might resize during this call because an Animate() on a screen might reschedule another animation etc */
   588 	TInt generation = iGeneration++;
   589 	TBool drew;
   590 	do
   591 		{
   592 		drew = EFalse;
   593 		const TInt count = iSchedule.Count();
   594 		for(TInt i=0; i<count; i++)
   595 			{
   596 			if(iSchedule[i].iSchedule)
   597 				{
   598 				scheduledCount++;
   599 				if(iSchedule[i].iGeneration == generation)
   600 					{
   601 					iSchedule[i].iSchedule = EFalse; // denote not to tick; it can be rescheduled in the subsequent call to Animate()
   602 					Animate(*(iSchedule[i].iScreen));
   603 					drew = ETrue;
   604 					drewCount++;
   605 					break; // go back to outer do while(drew)
   606 					}
   607 				}
   608 			}
   609 		} while(drew);
   610 	__DEBUG_ONLY(
   611 		TBool idleIsActive = iIdleInitiator->IsActive();
   612 		if(scheduledCount || !drewCount)
   613 			{
   614 			TBool rescheduled = EFalse;
   615 			for(TInt i=0; i<iSchedule.Count(); i++)
   616 				{
   617 				rescheduled |= iSchedule[i].iSchedule;
   618 				}
   619 			if(!rescheduled)
   620 				{
   621 				//__DEBUGGER();
   622 				}
   623 			})
   624 	}
   625 
   626 // CWindowServer \\\\\\\\\\\\\\\\\\\\\\\\\\\
   627 
   628 CWindowServer *CWindowServer::NewL()
   629 //
   630 // Create a new CWindowServer.
   631 //
   632 	{
   633 	CWindowServer* self = new(ELeave) CWindowServer();
   634 	CleanupStack::PushL(self);
   635 	self->ConstructL();
   636 	CleanupStack::Pop(self);
   637 	return self;
   638 	}
   639 
   640 CWindowServer::CWindowServer()
   641 //
   642 // Constructor.
   643 //
   644 	: CPolicyServer(EMainServerPriority,KWsServPolicy)
   645 	{
   646 	}
   647 
   648 CWindowServer::~CWindowServer()
   649 	{
   650 	iMemoryReleases.Reset();
   651 	WS_ASSERT_DEBUG(iDrawerMasterIndex.IsEmpty(), EWsPanicWsGraphic);
   652 	iDrawerMasterIndex.Close();
   653 	
   654 	delete iDefaultAnimationScheduler;
   655 	iDefaultAnimationScheduler = NULL; // might be called from clients during server destruction
   656 	delete iPluginManager;
   657 	}
   658 	
   659 void CWindowServer::ConstructL()
   660 	{
   661 	iPluginManager = CWsPluginManager::NewL(*this);
   662 	iDefaultAnimationScheduler = new(ELeave) CDefaultAnimationScheduler(*this);
   663 	iDefaultAnimationScheduler->ConstructL();
   664 	RegisterMemoryRelease(this);
   665 	}
   666 
   667 /** Creates a new client for this server. */
   668 CSession2* CWindowServer::NewSessionL(const TVersion& aVersion,const RMessage2& aMessage) const
   669 	{
   670 	TVersion v(KWservMajorVersionNumber,KWservMinorVersionNumber,KWservBuildVersionNumber);
   671 	if (User::QueryVersionSupported(v,aVersion)==EFalse)
   672 		User::Leave(KErrNotSupported);
   673 	RThread thread;
   674 	User::LeaveIfError(aMessage.Client(thread));
   675 	return(new(ELeave) CWsClient(thread));
   676 	}
   677 
   678 TInt CWindowServer::SessionCount()
   679 	{
   680 	iSessionIter.SetToFirst();
   681 	TInt count=0;
   682 	while(iSessionIter++)
   683 		count++;
   684 	return(count);
   685 	}
   686 
   687 const CWsGraphicDrawer* CWindowServer::ResolveGraphic(const TGraphicDrawerId& aId) const
   688 	{
   689 	return iDrawerMasterIndex.ResolveGraphic(aId);
   690 	}
   691 	
   692 void CWindowServer::Invalidate(const TGraphicDrawerId& aId)
   693 	{
   694 	AnimationScheduler()->Invalidate(aId);
   695 	}
   696 	
   697 TInt CWindowServer::ScreenCount() const
   698 	{
   699 	return CWsTop::NumberOfScreens();
   700 	}
   701 
   702 MWsScreen* CWindowServer::Screen(TInt aIndex)
   703 	{
   704 	if((aIndex >= 0) && (aIndex < ScreenCount()))
   705 		{
   706 		return CWsTop::Screen(aIndex);
   707 		}
   708 	return NULL;
   709 	}
   710 	
   711 const MWsScreen* CWindowServer::Screen(TInt aIndex) const
   712 	{
   713 	if((aIndex >= 0) && (aIndex < ScreenCount()))
   714 		{
   715 		return CWsTop::Screen(aIndex);
   716 		}
   717 	return NULL;
   718 	}
   719 	
   720 /**
   721 Custom Animation Scheduler
   722 */
   723 TBool CWindowServer::SetCustomAnimationScheduler(MWsAnimationScheduler* /*aScheduler*/)
   724 	{
   725 	return EFalse;
   726 	}
   727 
   728 TBool CWindowServer::HasCustomAnimationScheduler() const
   729 	{
   730 	return EFalse;
   731 	}
   732 
   733 TBool CWindowServer::ClearCustomAnimationScheduler(MWsAnimationScheduler* /*aCurrentScheduler*/)
   734 	{
   735 	return EFalse;
   736 	}
   737 	
   738 MWsAnimationScheduler* CWindowServer::AnimationScheduler()
   739 	{
   740 	return iDefaultAnimationScheduler;
   741 	}
   742 
   743 TInt CWindowServer::RegisterEventHandler(CWsGraphicDrawer* aDrawer, MWsEventHandler* aHandler, TUint32 aEventMask)
   744 	{
   745 	if (!aDrawer || !aHandler || aEventMask==0)
   746 		return KErrArgument;
   747 	TInt err = TWindowServerEvent::RegisterDrawerHandler(aDrawer, aEventMask);
   748 	if (err != KErrNone)
   749 		return err;
   750 	aDrawer->SetEventHandler(aHandler);
   751 	return KErrNone;
   752 	}
   753 	
   754 TInt CWindowServer::UnregisterEventHandler(CWsGraphicDrawer* aDrawer)
   755 	{
   756 	if (!aDrawer || (aDrawer && !aDrawer->HasEventHandler()))
   757 		return KErrArgument;
   758 	TInt err = TWindowServerEvent::UnregisterDrawerHandler(aDrawer);
   759 	if (err != KErrNone)
   760 		return err;
   761 	aDrawer->SetEventHandler(NULL);
   762 	return KErrNone;
   763 	}
   764 	
   765 TInt CWindowServer::RegisterWsEventHandler(MWsEventHandler* aHandler, TUint32 aEventMask)
   766 	{
   767 	if (!aHandler || aEventMask==0)
   768 		return KErrArgument;
   769 	return TWindowServerEvent::RegisterWsEventHandler(aHandler, aEventMask);
   770 	}
   771 	
   772 TInt CWindowServer::UnregisterWsEventHandler(MWsEventHandler* aHandler)
   773 	{
   774 	return TWindowServerEvent::UnregisterWsEventHandler(aHandler);
   775 	}
   776 	
   777 TInt CWindowServer::RegisterRawEventHandler(MEventHandler* aHandler)
   778 	{
   779 	if (!aHandler)
   780 		return KErrArgument;
   781 	TRAPD(err, TWindowServerEvent::PotentialEventHandlerL(1));
   782 	if (err != KErrNone)
   783 		return err;
   784 	TWindowServerEvent::AddEventHandler(aHandler);
   785 	return KErrNone;
   786 	}
   787 	
   788 void CWindowServer::UnregisterRawEventHandler(MEventHandler* aHandler)
   789 	{
   790 	TWindowServerEvent::RemoveEventHandler(aHandler);
   791 	TWindowServerEvent::PotentialEventHandlerL(-1); // can't leave for -1
   792 	}
   793 
   794 void CWindowServer::PostRawEvent(const TRawEvent & aEvent)
   795 	{
   796 	TWindowServerEvent::ProcessRawEvent(aEvent);
   797 	}
   798 	
   799 void CWindowServer::PostKeyEvent(const TKeyEvent & aEvent)
   800 	{
   801 	TWindowServerEvent::ProcessKeyEvent(aEvent, 0);
   802 	}
   803 	
   804 TAny* CWindowServer::ResolveObjectInterface(TUint aTypeId)
   805 	{
   806 	switch(aTypeId)
   807 		{
   808 		case MWsActiveSchedulerDebug::EWsObjectInterfaceId:
   809 			return static_cast<MWsActiveSchedulerDebug*>(CWsActiveScheduler::Static());
   810 		case MWsIniFile::EWsObjectInterfaceId:
   811 			return static_cast<MWsIniFile*>(WsIniFile);
   812 		case MWsRawEventServer::EWsObjectInterfaceId:
   813 			return static_cast<MWsRawEventServer*>(this);
   814 		}
   815 	
   816 	if (iPluginManager)
   817 		return iPluginManager->ResolveObjectInterface(aTypeId);
   818 	
   819 	return NULL;
   820 	}
   821 
   822 void CWindowServer::Log(TInt aPriority,const TDesC &aFmt,TInt aParam)
   823 	{
   824 	if (wsDebugLog)
   825 		{
   826 		wsDebugLog->MiscMessage(aPriority, aFmt, aParam);
   827 		}
   828 	}
   829 
   830 // CWsGraphicDrawer master index
   831 
   832 TInt CWindowServer::AddGraphicDrawer(CWsGraphicDrawer* aDrawer)
   833 	{
   834 	return iDrawerMasterIndex.Add(aDrawer);
   835 	}
   836 
   837 TInt CWindowServer::SwapGraphicDrawer(CWsGraphicDrawer* aDrawer)
   838 	{
   839 	return iDrawerMasterIndex.Swap(aDrawer);
   840 	}
   841 	
   842 TInt CWindowServer::RemoveGraphicDrawer(const TGraphicDrawerId& aId)
   843 	{
   844 	iDefaultAnimationScheduler->RemoveGraphicDrawer(aId);
   845 	return iDrawerMasterIndex.Remove(aId);
   846 	}
   847 
   848 TInt CWindowServer::RemoveAllGraphicDrawers(const MWsClient& aOwner)
   849 	{
   850 	return iDrawerMasterIndex.RemoveAll(aOwner);
   851 	}
   852 
   853 CWsPluginManager* CWindowServer::PluginManager()
   854 	{
   855 	return iPluginManager;
   856 	}
   857 
   858 TInt CWindowServer::RegisterMemoryRelease(MWsMemoryRelease * aMemoryRelease)
   859 	{
   860 	return iMemoryReleases.Append(aMemoryRelease);
   861 	}
   862 
   863 void CWindowServer::UnregisterMemoryRelease(MWsMemoryRelease * aMemoryRelease)
   864 	{
   865 	for (TInt i = iMemoryReleases.Count() - 1; i >= 0; --i)
   866 		{
   867 		if (iMemoryReleases[i] == aMemoryRelease)
   868 			{
   869 			iMemoryReleases.Remove(i);
   870 			break;
   871 			}
   872 		}
   873 	}
   874 
   875 TBool CWindowServer::ReleaseMemory(TMemoryReleaseLevel aLevel)
   876 	{
   877 	return CWsWindow::ReleaseMemory(aLevel);
   878 	}
   879 
   880 TBool CWindowServer::ReleaseMemory()
   881 	{
   882 	TBool released = EFalse;
   883 	for (TInt level = MWsMemoryRelease::ELow; !released && level <= MWsMemoryRelease::EHigh; ++level)
   884 		{
   885 		for (TInt i = iMemoryReleases.Count() - 1; !released && i >= 0; --i)
   886 			{
   887 			released = iMemoryReleases[i]->ReleaseMemory(static_cast<TMemoryReleaseLevel>(level));
   888 			}
   889 		}
   890 	return released;
   891 	}
   892