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