1.1 --- /dev/null Thu Jan 01 00:00:00 1970 +0000
1.2 +++ b/os/graphics/windowing/windowserver/nonnga/SERVER/SERVER.CPP Fri Jun 15 03:10:57 2012 +0200
1.3 @@ -0,0 +1,892 @@
1.4 +// Copyright (c) 1995-2009 Nokia Corporation and/or its subsidiary(-ies).
1.5 +// All rights reserved.
1.6 +// This component and the accompanying materials are made available
1.7 +// under the terms of "Eclipse Public License v1.0"
1.8 +// which accompanies this distribution, and is available
1.9 +// at the URL "http://www.eclipse.org/legal/epl-v10.html".
1.10 +//
1.11 +// Initial Contributors:
1.12 +// Nokia Corporation - initial contribution.
1.13 +//
1.14 +// Contributors:
1.15 +//
1.16 +// Description:
1.17 +// Window server 'server' class
1.18 +//
1.19 +//
1.20 +
1.21 +#include "server.h"
1.22 +#include "panics.h"
1.23 +#include "wstop.h"
1.24 +#include "EVENT.H"
1.25 +#include <bitdraw.h>
1.26 +#include <hal.h>
1.27 +#include "inifile.h"
1.28 +#include "wspluginmanager.h"
1.29 +
1.30 +GLREF_D CDebugLogBase *wsDebugLog;
1.31 +
1.32 +const TUint KRangeCount = 1;
1.33 +// We use a lot of 64 bit time calculations, but a periodic can only cope with signed 32 bit times
1.34 +// which gives them a limit of about 35 minutes.
1.35 +// Fortunately, our animtions are safe if redrawn early. Every half an hour isn't going to hurt.
1.36 +const TTimeIntervalMicroSeconds KHalfHour = 30 * 60 * 1000 * 1000;
1.37 +
1.38 +const TInt KWsServRanges[KRangeCount] =
1.39 + {
1.40 + 0
1.41 + };
1.42 +
1.43 +const TUint8 KElementsIndex[KRangeCount] =
1.44 + {
1.45 + CPolicyServer::EAlwaysPass,
1.46 + };
1.47 +
1.48 +const CPolicyServer::TPolicyElement KPolicyElements[] =
1.49 + {
1.50 + {_INIT_SECURITY_POLICY_C1(ECapabilityPowerMgmt), CPolicyServer::EFailClient},
1.51 + {_INIT_SECURITY_POLICY_C1(ECapabilitySwEvent), CPolicyServer::EFailClient},
1.52 + {_INIT_SECURITY_POLICY_C1(ECapabilityWriteDeviceData), CPolicyServer::EFailClient}
1.53 + };
1.54 +
1.55 +const CPolicyServer::TPolicy KWsServPolicy =
1.56 + {
1.57 + CPolicyServer::EAlwaysPass,
1.58 + KRangeCount,
1.59 + KWsServRanges,
1.60 + KElementsIndex,
1.61 + KPolicyElements
1.62 + };
1.63 +
1.64 +// CWindowServer::CDefaultAnimationScheduler \\\\\\\\\\\\\\\\\\\\\\\\\\\
1.65 +
1.66 +class CWindowServer::CDefaultAnimationScheduler: public CBase, public MWsAnimationScheduler
1.67 + {
1.68 +public:
1.69 + enum TInactivityBehaviour
1.70 + {
1.71 + EStopAnimation,
1.72 + EStopAllDrawing,
1.73 + EIgnore,
1.74 + };
1.75 + class CKickBack;
1.76 + CDefaultAnimationScheduler(MWsGraphicDrawerEnvironment& aEnv);
1.77 + ~CDefaultAnimationScheduler();
1.78 + void ConstructL(); //LeaveScan: member of nested class declaration
1.79 + // implementing MWsAnimationScheduler
1.80 + void ScheduleAnimation(MWsScreen& aScreen,const TTime& aWhen);
1.81 + void UnscheduleAnimation(MWsScreen& aScreen);
1.82 + void Invalidate(const TGraphicDrawerId& aId);
1.83 + void OnInactive();
1.84 + void OnActive();
1.85 + void ScheduleRedraw(MWsScreen& aScreen,const TTime& aWhen);
1.86 + void DoRedrawNow(MWsScreen& aScreen);
1.87 + void DoRedrawNow(MWsScreen& aScreen, MWsAnimationScheduler::MScreenUpdateObserver& aObserver);
1.88 + void ClearScreenUpdateObserver(const MWsAnimationScheduler::MScreenUpdateObserver& /*aObserver*/) {}
1.89 + TInt RemoveGraphicDrawer(const TGraphicDrawerId &aId);
1.90 +
1.91 +private:
1.92 + static TBool OnIdleCallBack(TAny* aAny);
1.93 + void OnIdleCallBack(TBool aForceRedraw);
1.94 + static TBool OnTickCallBack(TAny* aAny);
1.95 + void OnTickCallBack();
1.96 + static TBool OnKickBack(TAny* aAny);
1.97 + void OnKickBack();
1.98 +private:
1.99 + MWsGraphicDrawerEnvironment& iEnv;
1.100 + static const TInt64 KRedrawGrace;
1.101 + static const TInt64 KAnimationGrace;
1.102 + static const TInt64 KDsaAnimationGrace;
1.103 + CAsyncCallBack* iIdleInitiator;
1.104 + CPeriodic* iTick;
1.105 + CKickBack* iKickBack;
1.106 + struct TSchedule
1.107 + {
1.108 + MWsScreen* iScreen; // used as a unique index, searching with FindInUnsignedKeyOrder
1.109 + TBool iSchedule;
1.110 + TTime iWhen;
1.111 + TInt iGeneration;
1.112 + };
1.113 + RArray<TSchedule> iSchedule;
1.114 + TInt iGeneration;
1.115 + RArray<TGraphicDrawerId> iInvalidated;
1.116 + TBool iInvalidateAll; // if we could not add to iInvalidated, we have to instead redraw everything
1.117 + TTime iWhenDesired;
1.118 + TBool iInactive;
1.119 + TBool iInactiveDraws;
1.120 + TTime iExpectedTime;
1.121 + TBool iRedrawScheduled;
1.122 + TInt64 iRedrawGracePeriod;
1.123 + TInt64 iAnimationGracePeriod;
1.124 + TInt64 iDsaAnimationGracePeriod;
1.125 + TInactivityBehaviour iInactivityBehaviour;
1.126 + };
1.127 +
1.128 +TInt CWindowServer::CDefaultAnimationScheduler::RemoveGraphicDrawer(const TGraphicDrawerId &aId)
1.129 + {
1.130 + TInt index=iInvalidated.Find(aId);
1.131 + if (index!=KErrNotFound)
1.132 + iInvalidated.Remove(index);
1.133 + return index;
1.134 + }
1.135 +
1.136 +class CWindowServer::CDefaultAnimationScheduler::CKickBack: public CActive
1.137 + {
1.138 +public:
1.139 + CKickBack(const TCallBack& aCallBack);
1.140 + void ConstructL();
1.141 + void RequestKickBack();
1.142 + ~CKickBack();
1.143 +private:
1.144 + static TInt IdleThreadFunc(TAny* aAny);
1.145 + void Loop();
1.146 + // from CActive
1.147 + void RunL(); // fires when kicked back by the idle thread
1.148 + void DoCancel();
1.149 +private:
1.150 + RThread iWservThread;
1.151 + RThread iIdleThread;
1.152 + TRequestStatus iIdleStatus;
1.153 + TCallBack iCallBack;
1.154 + };
1.155 +
1.156 +CWindowServer::CDefaultAnimationScheduler::CKickBack::CKickBack(const TCallBack& aCallBack) :
1.157 +CActive(EPriorityNormal),
1.158 +iCallBack(aCallBack)
1.159 + {
1.160 + CActiveScheduler::Add(this);
1.161 + }
1.162 +
1.163 +void CWindowServer::CDefaultAnimationScheduler::CKickBack::ConstructL()
1.164 + {
1.165 + _LIT(KIdleThreadName,"NearlyIdleKickBack");
1.166 + const TInt KStackSize = 1024;
1.167 + User::LeaveIfError(iWservThread.Open(iWservThread.Id()));
1.168 + User::LeaveIfError(iIdleThread.Create(KIdleThreadName(),IdleThreadFunc,KStackSize,NULL,this));
1.169 + iIdleThread.SetPriority(EPriorityAbsoluteVeryLow);
1.170 + iIdleThread.Resume();
1.171 + }
1.172 +
1.173 +void CWindowServer::CDefaultAnimationScheduler::CKickBack::RequestKickBack()
1.174 + {
1.175 + if (!IsActive())
1.176 + {
1.177 + iStatus = KRequestPending;
1.178 + SetActive();
1.179 + TRequestStatus * status = &iIdleStatus;
1.180 + iIdleThread.RequestComplete(status, KErrNone);
1.181 + }
1.182 + }
1.183 +
1.184 +void CWindowServer::CDefaultAnimationScheduler::CKickBack::Loop()
1.185 + {
1.186 + FOREVER
1.187 + {
1.188 + // This is used here for performance reasons.
1.189 + User::WaitForRequest(iIdleStatus);
1.190 + iIdleStatus = KRequestPending;
1.191 + if (IsActive()&& (iStatus == KRequestPending))
1.192 + {
1.193 + TRequestStatus * status = &iStatus;
1.194 + iWservThread.RequestComplete(status,KErrNone);
1.195 + }
1.196 + }
1.197 + }
1.198 +
1.199 +void CWindowServer::CDefaultAnimationScheduler::CKickBack::RunL()
1.200 + {
1.201 + iCallBack.CallBack();
1.202 + }
1.203 +
1.204 +void CWindowServer::CDefaultAnimationScheduler::CKickBack::DoCancel()
1.205 + {
1.206 + }
1.207 +
1.208 +CWindowServer::CDefaultAnimationScheduler::CKickBack::~CKickBack()
1.209 + {
1.210 + Cancel();
1.211 + iWservThread.Close();
1.212 + iIdleThread.Kill(0);
1.213 + iIdleThread.Close();
1.214 + }
1.215 +
1.216 +TInt CWindowServer::CDefaultAnimationScheduler::CKickBack::IdleThreadFunc(TAny* aAny)
1.217 + {
1.218 + CKickBack* owner = reinterpret_cast<CKickBack*>(aAny);
1.219 + ASSERT(owner);
1.220 + if(owner)
1.221 + {
1.222 + owner->Loop();
1.223 + }
1.224 + return KErrNone;
1.225 + }
1.226 +
1.227 +// If using the default animation scheduler on a device, these two numbers may be worth tweaking in the inifile
1.228 +// However, both are maximum periods - wserv will go faster than either if nothing else is using the system.
1.229 +const TInt64 CWindowServer::CDefaultAnimationScheduler::KRedrawGrace = 0; // do redraws immediately
1.230 +const TInt64 CWindowServer::CDefaultAnimationScheduler::KAnimationGrace = 35000; // insist upon 35ms grace for other threads to run when animating
1.231 +const TInt64 CWindowServer::CDefaultAnimationScheduler::KDsaAnimationGrace = 35000; // insist upon 35ms grace for other threads to run when animating during DSA
1.232 +
1.233 +CWindowServer::CDefaultAnimationScheduler::CDefaultAnimationScheduler(MWsGraphicDrawerEnvironment& aEnv):
1.234 + iEnv(aEnv), iSchedule(1,_FOFF(TSchedule,iScreen))
1.235 + {
1.236 + }
1.237 +
1.238 +CWindowServer::CDefaultAnimationScheduler::~CDefaultAnimationScheduler()
1.239 + {
1.240 + iSchedule.Close();
1.241 + iInvalidated.Close();
1.242 + delete iKickBack;
1.243 + delete iIdleInitiator;
1.244 + delete iTick;
1.245 + }
1.246 +
1.247 +void CWindowServer::CDefaultAnimationScheduler::ConstructL()
1.248 + {
1.249 + _LIT(KOnInactive,"ONINACTIVE");
1.250 + _LIT(KStopAnimation,"STOPANIMATION");
1.251 + _LIT(KStopAllDrawing,"STOPALLDRAWING");
1.252 + _LIT(KIgnore,"IGNORE");
1.253 +
1.254 + TPtrC inactivityBehaviourString;
1.255 + WsIniFile->FindVar(KOnInactive,inactivityBehaviourString);
1.256 + if(inactivityBehaviourString.CompareF(KStopAnimation)==0)
1.257 + iInactivityBehaviour = EStopAnimation;
1.258 + else if(inactivityBehaviourString.CompareF(KStopAllDrawing)==0)
1.259 + iInactivityBehaviour = EStopAllDrawing;
1.260 + else if(inactivityBehaviourString.CompareF(KIgnore)==0)
1.261 + iInactivityBehaviour = EIgnore;
1.262 +
1.263 + _LIT(KRedrawGracePeriod, "REDRAWGRACEPERIOD");
1.264 + TInt tmp = KRedrawGrace;
1.265 + WsIniFile->FindVar(KRedrawGracePeriod, tmp);
1.266 + iRedrawGracePeriod = tmp;
1.267 +
1.268 + _LIT(KAnimationGracePeriod, "ANIMATIONGRACEPERIOD");
1.269 + tmp = KAnimationGrace;
1.270 + WsIniFile->FindVar(KAnimationGracePeriod, tmp);
1.271 + iAnimationGracePeriod = tmp;
1.272 +
1.273 + _LIT(KDsaAnimationGracePeriod, "DSAANIMATIONGRACEPERIOD");
1.274 + tmp = KDsaAnimationGrace;
1.275 + WsIniFile->FindVar(KDsaAnimationGracePeriod, tmp);
1.276 + iDsaAnimationGracePeriod = tmp;
1.277 +
1.278 + iIdleInitiator = new(ELeave) CAsyncCallBack(TCallBack(OnIdleCallBack,this),EWsGraphicAnimateAwaitIdlePriority);
1.279 + iTick = CPeriodic::NewL(EWsGraphicAnimatePriority);
1.280 +
1.281 + _LIT(KDisableIdleAnimation, "DISABLEIDLEANIMATION");
1.282 + if (!WsIniFile->FindVar(KDisableIdleAnimation))
1.283 + {
1.284 + iKickBack = new CKickBack(TCallBack(OnKickBack,this));
1.285 + iKickBack->ConstructL();
1.286 + }
1.287 + }
1.288 +
1.289 +void CWindowServer::CDefaultAnimationScheduler::Invalidate(const TGraphicDrawerId& aId)
1.290 + {
1.291 + if(!iInvalidateAll)
1.292 + {
1.293 + switch(iInvalidated.InsertInOrder(aId,TLinearOrder<TGraphicDrawerId>(TGraphicDrawerId::Compare)))
1.294 + {
1.295 + case KErrNone:
1.296 + break;
1.297 + case KErrAlreadyExists:
1.298 + break;
1.299 + default:
1.300 + iInvalidateAll = ETrue;
1.301 + iInvalidated.Reset();
1.302 + }
1.303 + }
1.304 + iIdleInitiator->CallBack();
1.305 + }
1.306 +
1.307 +void CWindowServer::CDefaultAnimationScheduler::OnInactive()
1.308 + {
1.309 + iInactive = ETrue;
1.310 + }
1.311 +
1.312 +void CWindowServer::CDefaultAnimationScheduler::OnActive()
1.313 + {
1.314 + iInactive = EFalse;
1.315 + if(iInactiveDraws)
1.316 + {
1.317 + iInactiveDraws = EFalse;
1.318 + iIdleInitiator->CallBack();
1.319 + }
1.320 + }
1.321 +
1.322 +void CWindowServer::CDefaultAnimationScheduler::ScheduleRedraw(MWsScreen& aScreen,const TTime& aWhen)
1.323 + {
1.324 + iRedrawScheduled = ETrue;
1.325 + ScheduleAnimation(aScreen, aWhen);
1.326 + }
1.327 +
1.328 +void CWindowServer::CDefaultAnimationScheduler::DoRedrawNow(MWsScreen& /*aScreen*/)
1.329 + {
1.330 + OnIdleCallBack(ETrue);
1.331 + }
1.332 +
1.333 +void CWindowServer::CDefaultAnimationScheduler::DoRedrawNow(MWsScreen& /*aScreen*/, MWsAnimationScheduler::MScreenUpdateObserver& aObserver)
1.334 + {
1.335 + OnIdleCallBack(ETrue);
1.336 + aObserver.ScreenUpdateComplete(KErrNone);
1.337 + }
1.338 +
1.339 +void CWindowServer::CDefaultAnimationScheduler::ScheduleAnimation(MWsScreen& aScreen,const TTime& aWhen)
1.340 + {
1.341 + TSchedule schedule;
1.342 + schedule.iScreen = &aScreen;
1.343 + schedule.iSchedule = ETrue;
1.344 + schedule.iWhen = aWhen;
1.345 + schedule.iGeneration = iGeneration;
1.346 + TBool ok = EFalse;
1.347 + const TInt idx = iSchedule.FindInUnsignedKeyOrder(schedule);
1.348 + if(0 <= idx)
1.349 + {
1.350 + if(iSchedule[idx].iSchedule)
1.351 + {
1.352 + if(iSchedule[idx].iWhen > aWhen)
1.353 + {
1.354 + iSchedule[idx].iWhen = aWhen;
1.355 + }
1.356 + }
1.357 + else
1.358 + {
1.359 + iSchedule[idx] = schedule;
1.360 + }
1.361 + iSchedule[idx].iGeneration = iGeneration;
1.362 + ok = ETrue;
1.363 + }
1.364 + else
1.365 + {
1.366 + ok = (KErrNone == iSchedule.InsertInUnsignedKeyOrder(schedule));
1.367 + }
1.368 + if(ok)
1.369 + {
1.370 + //If the animation runs at very high rate which exceeds the rate WSERV can
1.371 + //perform the rendering, it is possible that WSERV animation loop will monopolize
1.372 + //processor time.
1.373 + User::After(0); // to yeild from the animation loop
1.374 +
1.375 + iIdleInitiator->CallBack();
1.376 + }
1.377 + }
1.378 +
1.379 +void CWindowServer::CDefaultAnimationScheduler::UnscheduleAnimation(MWsScreen& aScreen)
1.380 + {
1.381 + TSchedule schedule;
1.382 + schedule.iScreen = &aScreen;
1.383 + const TInt idx = iSchedule.FindInUnsignedKeyOrder(schedule);
1.384 + if(0 <= idx)
1.385 + {
1.386 + iSchedule[idx].iSchedule = EFalse;
1.387 + }
1.388 + }
1.389 +
1.390 +TBool CWindowServer::CDefaultAnimationScheduler::OnIdleCallBack(TAny* aAny)
1.391 + {
1.392 + CDefaultAnimationScheduler* self = reinterpret_cast<CDefaultAnimationScheduler*>(aAny);
1.393 + ASSERT(self);
1.394 + if(self)
1.395 + {
1.396 + self->OnIdleCallBack(EFalse);
1.397 + }
1.398 + return EFalse; //ignored by caller
1.399 + }
1.400 +
1.401 +void CWindowServer::CDefaultAnimationScheduler::OnIdleCallBack(TBool aForceRedraw)
1.402 + {
1.403 + TBool wasActive = EFalse;
1.404 + if (iTick->IsActive())
1.405 + {
1.406 + wasActive = ETrue;
1.407 + iTick->Cancel(); // stop ticker, as we'll reschedule if necessary
1.408 + }
1.409 + if (aForceRedraw)
1.410 + {
1.411 + // Don't need to update iExpectedTime as we will not schedule a tick.
1.412 + // Don't need to update iWhenDesired as we will not schedule a kick-back.
1.413 + OnTickCallBack();
1.414 + return;
1.415 + }
1.416 + TBool tick = (iInvalidateAll || iInvalidated.Count());
1.417 + TTimeIntervalMicroSeconds next = 0LL;
1.418 + const TInt count = iSchedule.Count();
1.419 + TTime now;
1.420 + now.UniversalTime();
1.421 + if(count)
1.422 + {
1.423 + // work out the next wanted tick
1.424 + TBool animTick = EFalse;
1.425 + for(TInt i=0; i<count; i++)
1.426 + {
1.427 + if(iSchedule[i].iSchedule)
1.428 + {
1.429 + const TTime when = iSchedule[i].iWhen;
1.430 + const TTimeIntervalMicroSeconds fromNow = when.MicroSecondsFrom(now);
1.431 + if(!animTick || (fromNow < next))
1.432 + {
1.433 + animTick = ETrue;
1.434 + tick = ETrue;
1.435 + next = fromNow;
1.436 + iWhenDesired = when;
1.437 + }
1.438 + }
1.439 + }
1.440 + }
1.441 + if(tick)
1.442 + {
1.443 + TInt64 grace;
1.444 + if (iRedrawScheduled)
1.445 + grace = iRedrawGracePeriod;
1.446 + else if (EFalse) // DSA active
1.447 + grace = iDsaAnimationGracePeriod;
1.448 + else
1.449 + grace = iAnimationGracePeriod;
1.450 + if (wasActive)
1.451 + {
1.452 + TInt64 minimum = iExpectedTime.MicroSecondsFrom(now).Int64();
1.453 + if (minimum >= 0 && grace > minimum)
1.454 + grace = minimum;
1.455 + }
1.456 +
1.457 + if (next.Int64() <= 0)
1.458 + next = 0LL ; // No kickback/tick is needed. Make sure next == default grace period.
1.459 +
1.460 + if(next < grace)
1.461 + {
1.462 + next = grace;
1.463 + if (iKickBack)
1.464 + iKickBack->RequestKickBack();
1.465 + }
1.466 + else if (next > KHalfHour)
1.467 + next = KHalfHour;
1.468 +
1.469 + iExpectedTime=now + next;
1.470 + if (next.Int64() > 0)
1.471 + {
1.472 + iTick->Start(next.Int64(),0,TCallBack(OnTickCallBack,this));
1.473 + }
1.474 + else
1.475 + {
1.476 + OnTickCallBack(); // scheduling for 0 doesn't actually execute immediately
1.477 + }
1.478 + }
1.479 + }
1.480 +
1.481 +TBool CWindowServer::CDefaultAnimationScheduler::OnKickBack(TAny* aAny)
1.482 + {
1.483 + CDefaultAnimationScheduler* self = reinterpret_cast<CDefaultAnimationScheduler*>(aAny);
1.484 + ASSERT(self);
1.485 + if(self)
1.486 + {
1.487 + self->OnKickBack();
1.488 + }
1.489 + return EFalse; //ignored by caller
1.490 + }
1.491 +
1.492 +void CWindowServer::CDefaultAnimationScheduler::OnKickBack()
1.493 + {
1.494 + if (iTick->IsActive())
1.495 + {
1.496 + iTick->Cancel();
1.497 + TTime now;
1.498 + now.UniversalTime();
1.499 +
1.500 + TTimeIntervalMicroSeconds fromNow = iWhenDesired.MicroSecondsFrom(now);
1.501 +
1.502 + if (fromNow < 0)
1.503 + {
1.504 + OnTickCallBack();
1.505 + }
1.506 + else
1.507 + {
1.508 + if (fromNow > KHalfHour)
1.509 + fromNow = KHalfHour;
1.510 + iTick->Start(fromNow.Int64(),0,TCallBack(OnTickCallBack, this));
1.511 + }
1.512 + }
1.513 + }
1.514 +
1.515 +TBool CWindowServer::CDefaultAnimationScheduler::OnTickCallBack(TAny* aAny)
1.516 + {
1.517 + CDefaultAnimationScheduler* self = reinterpret_cast<CDefaultAnimationScheduler*>(aAny);
1.518 + ASSERT(self);
1.519 + if(self)
1.520 + {
1.521 + self->OnTickCallBack();
1.522 + }
1.523 + return EFalse;
1.524 + }
1.525 +
1.526 +void CWindowServer::CDefaultAnimationScheduler::OnTickCallBack()
1.527 + {
1.528 + iTick->Cancel();
1.529 +
1.530 + switch(iInactivityBehaviour)
1.531 + {
1.532 + case EStopAnimation :
1.533 + // only client redraws if inactive. server side drawing stopped.
1.534 + if(iInactive && !iRedrawScheduled)
1.535 + {
1.536 + iInactiveDraws = ETrue;
1.537 + return;
1.538 + }
1.539 + break;
1.540 + case EStopAllDrawing :
1.541 + // if screen off, stop both client and server side drawing.
1.542 + if(iInactive)
1.543 + {
1.544 + iInactiveDraws = ETrue;
1.545 + return;
1.546 + }
1.547 + break;
1.548 + case EIgnore :
1.549 + default :
1.550 + // ignore inactivity and draw as normal
1.551 + break;
1.552 + }
1.553 +
1.554 + iRedrawScheduled = EFalse;
1.555 +
1.556 + TTime now;
1.557 + now.UniversalTime();
1.558 + CWsActiveScheduler::Static()->AccumReclaimedIdleTime(Max(0LL,now.MicroSecondsFrom(iExpectedTime).Int64()));
1.559 +
1.560 + TInt drewCount = 0;
1.561 + TInt scheduledCount = 0;
1.562 + /* first redraw any screens that are affected by invalidated graphic IDs */
1.563 + if(iInvalidateAll || iInvalidated.Count())
1.564 + {
1.565 + // cancel idle callback if it's already scheduled as we're going to redraw all invalidated
1.566 + // request at this point and clear invalidated array
1.567 + if (iIdleInitiator->IsActive())
1.568 + iIdleInitiator->Cancel();
1.569 + const TArray<TGraphicDrawerId> invalidArray = iInvalidated.Array();
1.570 + const TInt screenCount = iEnv.ScreenCount();
1.571 + for(TInt i=0; i<screenCount; i++)
1.572 + {
1.573 + MWsScreen* screen = iEnv.Screen(i);
1.574 + if(screen)
1.575 + {
1.576 + drewCount++;
1.577 + if(iInvalidateAll)
1.578 + {
1.579 + Redraw(*screen);
1.580 + }
1.581 + else
1.582 + {
1.583 + RedrawInvalid(*screen,invalidArray);
1.584 + }
1.585 + }
1.586 + }
1.587 + iInvalidateAll = EFalse;
1.588 + iInvalidated.Reset();
1.589 + }
1.590 + /* iSchedule might resize during this call because an Animate() on a screen might reschedule another animation etc */
1.591 + TInt generation = iGeneration++;
1.592 + TBool drew;
1.593 + do
1.594 + {
1.595 + drew = EFalse;
1.596 + const TInt count = iSchedule.Count();
1.597 + for(TInt i=0; i<count; i++)
1.598 + {
1.599 + if(iSchedule[i].iSchedule)
1.600 + {
1.601 + scheduledCount++;
1.602 + if(iSchedule[i].iGeneration == generation)
1.603 + {
1.604 + iSchedule[i].iSchedule = EFalse; // denote not to tick; it can be rescheduled in the subsequent call to Animate()
1.605 + Animate(*(iSchedule[i].iScreen));
1.606 + drew = ETrue;
1.607 + drewCount++;
1.608 + break; // go back to outer do while(drew)
1.609 + }
1.610 + }
1.611 + }
1.612 + } while(drew);
1.613 + __DEBUG_ONLY(
1.614 + TBool idleIsActive = iIdleInitiator->IsActive();
1.615 + if(scheduledCount || !drewCount)
1.616 + {
1.617 + TBool rescheduled = EFalse;
1.618 + for(TInt i=0; i<iSchedule.Count(); i++)
1.619 + {
1.620 + rescheduled |= iSchedule[i].iSchedule;
1.621 + }
1.622 + if(!rescheduled)
1.623 + {
1.624 + //__DEBUGGER();
1.625 + }
1.626 + })
1.627 + }
1.628 +
1.629 +// CWindowServer \\\\\\\\\\\\\\\\\\\\\\\\\\\
1.630 +
1.631 +CWindowServer *CWindowServer::NewL()
1.632 +//
1.633 +// Create a new CWindowServer.
1.634 +//
1.635 + {
1.636 + CWindowServer* self = new(ELeave) CWindowServer();
1.637 + CleanupStack::PushL(self);
1.638 + self->ConstructL();
1.639 + CleanupStack::Pop(self);
1.640 + return self;
1.641 + }
1.642 +
1.643 +CWindowServer::CWindowServer()
1.644 +//
1.645 +// Constructor.
1.646 +//
1.647 + : CPolicyServer(EMainServerPriority,KWsServPolicy)
1.648 + {
1.649 + }
1.650 +
1.651 +CWindowServer::~CWindowServer()
1.652 + {
1.653 + iMemoryReleases.Reset();
1.654 + WS_ASSERT_DEBUG(iDrawerMasterIndex.IsEmpty(), EWsPanicWsGraphic);
1.655 + iDrawerMasterIndex.Close();
1.656 +
1.657 + delete iDefaultAnimationScheduler;
1.658 + iDefaultAnimationScheduler = NULL; // might be called from clients during server destruction
1.659 + delete iPluginManager;
1.660 + }
1.661 +
1.662 +void CWindowServer::ConstructL()
1.663 + {
1.664 + iPluginManager = CWsPluginManager::NewL(*this);
1.665 + iDefaultAnimationScheduler = new(ELeave) CDefaultAnimationScheduler(*this);
1.666 + iDefaultAnimationScheduler->ConstructL();
1.667 + RegisterMemoryRelease(this);
1.668 + }
1.669 +
1.670 +/** Creates a new client for this server. */
1.671 +CSession2* CWindowServer::NewSessionL(const TVersion& aVersion,const RMessage2& aMessage) const
1.672 + {
1.673 + TVersion v(KWservMajorVersionNumber,KWservMinorVersionNumber,KWservBuildVersionNumber);
1.674 + if (User::QueryVersionSupported(v,aVersion)==EFalse)
1.675 + User::Leave(KErrNotSupported);
1.676 + RThread thread;
1.677 + User::LeaveIfError(aMessage.Client(thread));
1.678 + return(new(ELeave) CWsClient(thread));
1.679 + }
1.680 +
1.681 +TInt CWindowServer::SessionCount()
1.682 + {
1.683 + iSessionIter.SetToFirst();
1.684 + TInt count=0;
1.685 + while(iSessionIter++)
1.686 + count++;
1.687 + return(count);
1.688 + }
1.689 +
1.690 +const CWsGraphicDrawer* CWindowServer::ResolveGraphic(const TGraphicDrawerId& aId) const
1.691 + {
1.692 + return iDrawerMasterIndex.ResolveGraphic(aId);
1.693 + }
1.694 +
1.695 +void CWindowServer::Invalidate(const TGraphicDrawerId& aId)
1.696 + {
1.697 + AnimationScheduler()->Invalidate(aId);
1.698 + }
1.699 +
1.700 +TInt CWindowServer::ScreenCount() const
1.701 + {
1.702 + return CWsTop::NumberOfScreens();
1.703 + }
1.704 +
1.705 +MWsScreen* CWindowServer::Screen(TInt aIndex)
1.706 + {
1.707 + if((aIndex >= 0) && (aIndex < ScreenCount()))
1.708 + {
1.709 + return CWsTop::Screen(aIndex);
1.710 + }
1.711 + return NULL;
1.712 + }
1.713 +
1.714 +const MWsScreen* CWindowServer::Screen(TInt aIndex) const
1.715 + {
1.716 + if((aIndex >= 0) && (aIndex < ScreenCount()))
1.717 + {
1.718 + return CWsTop::Screen(aIndex);
1.719 + }
1.720 + return NULL;
1.721 + }
1.722 +
1.723 +/**
1.724 +Custom Animation Scheduler
1.725 +*/
1.726 +TBool CWindowServer::SetCustomAnimationScheduler(MWsAnimationScheduler* /*aScheduler*/)
1.727 + {
1.728 + return EFalse;
1.729 + }
1.730 +
1.731 +TBool CWindowServer::HasCustomAnimationScheduler() const
1.732 + {
1.733 + return EFalse;
1.734 + }
1.735 +
1.736 +TBool CWindowServer::ClearCustomAnimationScheduler(MWsAnimationScheduler* /*aCurrentScheduler*/)
1.737 + {
1.738 + return EFalse;
1.739 + }
1.740 +
1.741 +MWsAnimationScheduler* CWindowServer::AnimationScheduler()
1.742 + {
1.743 + return iDefaultAnimationScheduler;
1.744 + }
1.745 +
1.746 +TInt CWindowServer::RegisterEventHandler(CWsGraphicDrawer* aDrawer, MWsEventHandler* aHandler, TUint32 aEventMask)
1.747 + {
1.748 + if (!aDrawer || !aHandler || aEventMask==0)
1.749 + return KErrArgument;
1.750 + TInt err = TWindowServerEvent::RegisterDrawerHandler(aDrawer, aEventMask);
1.751 + if (err != KErrNone)
1.752 + return err;
1.753 + aDrawer->SetEventHandler(aHandler);
1.754 + return KErrNone;
1.755 + }
1.756 +
1.757 +TInt CWindowServer::UnregisterEventHandler(CWsGraphicDrawer* aDrawer)
1.758 + {
1.759 + if (!aDrawer || (aDrawer && !aDrawer->HasEventHandler()))
1.760 + return KErrArgument;
1.761 + TInt err = TWindowServerEvent::UnregisterDrawerHandler(aDrawer);
1.762 + if (err != KErrNone)
1.763 + return err;
1.764 + aDrawer->SetEventHandler(NULL);
1.765 + return KErrNone;
1.766 + }
1.767 +
1.768 +TInt CWindowServer::RegisterWsEventHandler(MWsEventHandler* aHandler, TUint32 aEventMask)
1.769 + {
1.770 + if (!aHandler || aEventMask==0)
1.771 + return KErrArgument;
1.772 + return TWindowServerEvent::RegisterWsEventHandler(aHandler, aEventMask);
1.773 + }
1.774 +
1.775 +TInt CWindowServer::UnregisterWsEventHandler(MWsEventHandler* aHandler)
1.776 + {
1.777 + return TWindowServerEvent::UnregisterWsEventHandler(aHandler);
1.778 + }
1.779 +
1.780 +TInt CWindowServer::RegisterRawEventHandler(MEventHandler* aHandler)
1.781 + {
1.782 + if (!aHandler)
1.783 + return KErrArgument;
1.784 + TRAPD(err, TWindowServerEvent::PotentialEventHandlerL(1));
1.785 + if (err != KErrNone)
1.786 + return err;
1.787 + TWindowServerEvent::AddEventHandler(aHandler);
1.788 + return KErrNone;
1.789 + }
1.790 +
1.791 +void CWindowServer::UnregisterRawEventHandler(MEventHandler* aHandler)
1.792 + {
1.793 + TWindowServerEvent::RemoveEventHandler(aHandler);
1.794 + TWindowServerEvent::PotentialEventHandlerL(-1); // can't leave for -1
1.795 + }
1.796 +
1.797 +void CWindowServer::PostRawEvent(const TRawEvent & aEvent)
1.798 + {
1.799 + TWindowServerEvent::ProcessRawEvent(aEvent);
1.800 + }
1.801 +
1.802 +void CWindowServer::PostKeyEvent(const TKeyEvent & aEvent)
1.803 + {
1.804 + TWindowServerEvent::ProcessKeyEvent(aEvent, 0);
1.805 + }
1.806 +
1.807 +TAny* CWindowServer::ResolveObjectInterface(TUint aTypeId)
1.808 + {
1.809 + switch(aTypeId)
1.810 + {
1.811 + case MWsActiveSchedulerDebug::EWsObjectInterfaceId:
1.812 + return static_cast<MWsActiveSchedulerDebug*>(CWsActiveScheduler::Static());
1.813 + case MWsIniFile::EWsObjectInterfaceId:
1.814 + return static_cast<MWsIniFile*>(WsIniFile);
1.815 + case MWsRawEventServer::EWsObjectInterfaceId:
1.816 + return static_cast<MWsRawEventServer*>(this);
1.817 + }
1.818 +
1.819 + if (iPluginManager)
1.820 + return iPluginManager->ResolveObjectInterface(aTypeId);
1.821 +
1.822 + return NULL;
1.823 + }
1.824 +
1.825 +void CWindowServer::Log(TInt aPriority,const TDesC &aFmt,TInt aParam)
1.826 + {
1.827 + if (wsDebugLog)
1.828 + {
1.829 + wsDebugLog->MiscMessage(aPriority, aFmt, aParam);
1.830 + }
1.831 + }
1.832 +
1.833 +// CWsGraphicDrawer master index
1.834 +
1.835 +TInt CWindowServer::AddGraphicDrawer(CWsGraphicDrawer* aDrawer)
1.836 + {
1.837 + return iDrawerMasterIndex.Add(aDrawer);
1.838 + }
1.839 +
1.840 +TInt CWindowServer::SwapGraphicDrawer(CWsGraphicDrawer* aDrawer)
1.841 + {
1.842 + return iDrawerMasterIndex.Swap(aDrawer);
1.843 + }
1.844 +
1.845 +TInt CWindowServer::RemoveGraphicDrawer(const TGraphicDrawerId& aId)
1.846 + {
1.847 + iDefaultAnimationScheduler->RemoveGraphicDrawer(aId);
1.848 + return iDrawerMasterIndex.Remove(aId);
1.849 + }
1.850 +
1.851 +TInt CWindowServer::RemoveAllGraphicDrawers(const MWsClient& aOwner)
1.852 + {
1.853 + return iDrawerMasterIndex.RemoveAll(aOwner);
1.854 + }
1.855 +
1.856 +CWsPluginManager* CWindowServer::PluginManager()
1.857 + {
1.858 + return iPluginManager;
1.859 + }
1.860 +
1.861 +TInt CWindowServer::RegisterMemoryRelease(MWsMemoryRelease * aMemoryRelease)
1.862 + {
1.863 + return iMemoryReleases.Append(aMemoryRelease);
1.864 + }
1.865 +
1.866 +void CWindowServer::UnregisterMemoryRelease(MWsMemoryRelease * aMemoryRelease)
1.867 + {
1.868 + for (TInt i = iMemoryReleases.Count() - 1; i >= 0; --i)
1.869 + {
1.870 + if (iMemoryReleases[i] == aMemoryRelease)
1.871 + {
1.872 + iMemoryReleases.Remove(i);
1.873 + break;
1.874 + }
1.875 + }
1.876 + }
1.877 +
1.878 +TBool CWindowServer::ReleaseMemory(TMemoryReleaseLevel aLevel)
1.879 + {
1.880 + return CWsWindow::ReleaseMemory(aLevel);
1.881 + }
1.882 +
1.883 +TBool CWindowServer::ReleaseMemory()
1.884 + {
1.885 + TBool released = EFalse;
1.886 + for (TInt level = MWsMemoryRelease::ELow; !released && level <= MWsMemoryRelease::EHigh; ++level)
1.887 + {
1.888 + for (TInt i = iMemoryReleases.Count() - 1; !released && i >= 0; --i)
1.889 + {
1.890 + released = iMemoryReleases[i]->ReleaseMemory(static_cast<TMemoryReleaseLevel>(level));
1.891 + }
1.892 + }
1.893 + return released;
1.894 + }
1.895 +