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