1.1 --- /dev/null Thu Jan 01 00:00:00 1970 +0000
1.2 +++ b/os/graphics/windowing/windowserver/nga/SERVER/SERVER.CPP Fri Jun 15 03:10:57 2012 +0200
1.3 @@ -0,0 +1,1081 @@
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 +#include "graphics/windowserverconstants.h"
1.30 +
1.31 +GLREF_D CDebugLogBase *wsDebugLog;
1.32 +
1.33 +const TUint KRangeCount = 1;
1.34 +// We use a lot of 64 bit time calculations, but a periodic can only cope with signed 32 bit times
1.35 +// which gives them a limit of about 35 minutes.
1.36 +// Fortunately, our animtions are safe if redrawn early. Every half an hour isn't going to hurt.
1.37 +const TTimeIntervalMicroSeconds KHalfHour = 30 * 60 * 1000 * 1000;
1.38 +
1.39 +const TInt KWsServRanges[KRangeCount] =
1.40 + {
1.41 + 0
1.42 + };
1.43 +
1.44 +const TUint8 KElementsIndex[KRangeCount] =
1.45 + {
1.46 + CPolicyServer::EAlwaysPass,
1.47 + };
1.48 +
1.49 +const CPolicyServer::TPolicyElement KPolicyElements[] =
1.50 + {
1.51 + {_INIT_SECURITY_POLICY_C1(ECapabilityPowerMgmt), CPolicyServer::EFailClient},
1.52 + {_INIT_SECURITY_POLICY_C1(ECapabilitySwEvent), CPolicyServer::EFailClient},
1.53 + {_INIT_SECURITY_POLICY_C1(ECapabilityWriteDeviceData), CPolicyServer::EFailClient}
1.54 + };
1.55 +
1.56 +const CPolicyServer::TPolicy KWsServPolicy =
1.57 + {
1.58 + CPolicyServer::EAlwaysPass,
1.59 + KRangeCount,
1.60 + KWsServRanges,
1.61 + KElementsIndex,
1.62 + KPolicyElements
1.63 + };
1.64 +
1.65 +// CWindowServer::CDefaultAnimationScheduler \\\\\\\\\\\\\\\\\\\\\\\\\\\
1.66 +
1.67 +class CWindowServer::CDefaultAnimationScheduler: public CBase, public MWsAnimationScheduler
1.68 + {
1.69 + // Associates a screen number with a CActiveSchedulerWait intance. This is used to
1.70 + // achieve synchronous update completion on a specific screen.
1.71 + class CScreenUpdateWait : public CActiveSchedulerWait
1.72 + {
1.73 + public:
1.74 + CScreenUpdateWait(TInt aScreenNumber) : iScreenNumber (aScreenNumber) {}
1.75 + TInt iScreenNumber;
1.76 + };
1.77 +
1.78 + struct TScreenUpdateDetails
1.79 + {
1.80 + CWindowServer::CDefaultAnimationScheduler* iScheduler;
1.81 + TInt iScreenNumber;
1.82 + };
1.83 +
1.84 + struct TSchedule
1.85 + {
1.86 + MWsScreen* iScreen; // used as a unique index, searching with FindInUnsignedKeyOrder
1.87 + TInt iScreenNumber;
1.88 + TBool iScheduled;
1.89 + TTime iWhen;
1.90 + TBool iRedraw;
1.91 + };
1.92 +public:
1.93 + enum TInactivityBehaviour
1.94 + {
1.95 + EStopAnimation,
1.96 + EStopAllDrawing,
1.97 + EIgnore,
1.98 + };
1.99 + class CScreenState;
1.100 +
1.101 + CDefaultAnimationScheduler(MWsGraphicDrawerEnvironment& aEnv);
1.102 + ~CDefaultAnimationScheduler();
1.103 + void ConstructL(); //LeaveScan: member of nested class declaration
1.104 + // implementing MWsAnimationScheduler
1.105 + void ScheduleAnimation(MWsScreen& aScreen,const TTime& aWhen);
1.106 + void UnscheduleAnimation(MWsScreen& aScreen);
1.107 + void Invalidate(const TGraphicDrawerId& aId);
1.108 + void OnInactive();
1.109 + void OnActive();
1.110 + void ScheduleRedraw(MWsScreen& aScreen,const TTime& aWhen);
1.111 + void DoRedrawNow(MWsScreen& aScreen);
1.112 + void DoRedrawNow(MWsScreen& aScreen, MWsAnimationScheduler::MScreenUpdateObserver& aObserver);
1.113 + void ClearScreenUpdateObserver(const MWsAnimationScheduler::MScreenUpdateObserver& aObserver);
1.114 +private:
1.115 + static TBool OnIdleCallBack(TAny* aAny);
1.116 + void ScheduleUpdate (TInt aScreenNumber, TBool aForce);
1.117 + void OnIdleCallBack(TBool aForce);
1.118 + static TBool InvokeDueAnimation(TAny* aAny);
1.119 + void RedrawAllInvalidatedRegions (TInt aScreen);
1.120 + TSchedule* GetScheduledScreenUpdate(TInt aScreen);
1.121 + void InvokeDueAnimation(TInt aScreen);
1.122 + void ProcessUpdateCompletion (TInt aScreenNumber);
1.123 + TInt ScreenNumber(MWsScreen& aScreen) const;
1.124 + TTimeIntervalMicroSeconds GetDueDelta (TBool aForceRedraw, TSchedule* aScheduledUpdate);
1.125 +private:
1.126 + RPointerArray<CScreenUpdateWait> iRedrawWaitLoop;
1.127 + RPointerArray<CScreenState> iScreenState;
1.128 + MWsGraphicDrawerEnvironment& iEnv;
1.129 + static const TInt64 KRedrawGrace;
1.130 + static const TInt64 KAnimationGrace;
1.131 + CAsyncCallBack* iIdleInitiator;
1.132 + TBool iInRedrawNow;
1.133 + RArray<TSchedule> iSchedule;
1.134 + TBool iInactive;
1.135 + TBool iInactiveDraws;
1.136 + TBool iRedrawScheduled;
1.137 + TInt64 iRedrawGracePeriod;
1.138 + TInt64 iAnimationGracePeriod;
1.139 + TInactivityBehaviour iInactivityBehaviour;
1.140 + };
1.141 +
1.142 +class CWindowServer::CDefaultAnimationScheduler::CScreenState : public CActive
1.143 + {
1.144 +public:
1.145 + static CScreenState* NewL (CDefaultAnimationScheduler* aScheduler, TInt aScreenOrdinal);
1.146 + ~CScreenState();
1.147 + void SetActive ();
1.148 +
1.149 + void WaitForRedraws(MWsAnimationScheduler::MScreenUpdateObserver& aObserver, TInt aNumRedraws);
1.150 + void ClearScreenUpdateObserver(const MWsAnimationScheduler::MScreenUpdateObserver& aObserver);
1.151 + CPeriodic* iUpdateOn;
1.152 + TTime iExpectedTickTime;
1.153 + TScreenUpdateDetails iScreenUpdateDetails;
1.154 + RArray<TGraphicDrawerId> iInvalidated;
1.155 + TBool iInvalidateAll;
1.156 +
1.157 +private:
1.158 + CScreenState (CDefaultAnimationScheduler* aScheduler, TInt aScreenOrdinal);
1.159 + void ConstructL ();
1.160 + void ReleaseRemainingClients();
1.161 + void ReleaseClientsWaitingFor(TUint aCurrentFrame);
1.162 +
1.163 + void RunL();
1.164 + void DoCancel()
1.165 + {
1.166 + TRequestStatus* tmpTRS = &iStatus;
1.167 + User::RequestComplete(tmpTRS, KErrNone);
1.168 + };
1.169 + class TWaitingClient
1.170 + {
1.171 + public:
1.172 + TWaitingClient(MWsAnimationScheduler::MScreenUpdateObserver& aObserver, TInt aTargetFrame)
1.173 + : iObserver(aObserver), iTargetFrame(aTargetFrame)
1.174 + {
1.175 + }
1.176 + MWsAnimationScheduler::MScreenUpdateObserver& iObserver;
1.177 + TUint iTargetFrame;
1.178 + };
1.179 + TUint iFrameCount;
1.180 + RArray<TWaitingClient> iWaitingClients;
1.181 + };
1.182 +
1.183 +// If using the default animation scheduler on a device, these two numbers may be worth tweaking in the inifile
1.184 +// However, both are maximum periods - wserv will go faster than either if nothing else is using the system.
1.185 +const TInt64 CWindowServer::CDefaultAnimationScheduler::KRedrawGrace = 0; // do redraws immediately
1.186 +const TInt64 CWindowServer::CDefaultAnimationScheduler::KAnimationGrace = 0; // do animation redraws immediately
1.187 +
1.188 +CWindowServer::CDefaultAnimationScheduler::CDefaultAnimationScheduler(MWsGraphicDrawerEnvironment& aEnv):
1.189 + iEnv(aEnv), iSchedule(1,_FOFF(TSchedule,iScreen))
1.190 + {
1.191 + }
1.192 +
1.193 +CWindowServer::CDefaultAnimationScheduler::~CDefaultAnimationScheduler()
1.194 + {
1.195 + iSchedule.Close();
1.196 + delete iIdleInitiator;
1.197 +
1.198 + TInt screenCount = iScreenState.Count();
1.199 + for (TInt ii = 0; ii < screenCount; ii++)
1.200 + delete iScreenState[ii];
1.201 +
1.202 + iScreenState.Close();
1.203 +
1.204 + // Destroy the redraw wait loops associated each screen, and close
1.205 + // the associated Array objects.
1.206 + TInt waitLoopCount = iRedrawWaitLoop.Count();
1.207 + for (TInt waitLoop = 0; waitLoop < waitLoopCount; waitLoop++)
1.208 + delete iRedrawWaitLoop[waitLoop];
1.209 + iRedrawWaitLoop.Close();
1.210 + }
1.211 +
1.212 +void CWindowServer::CDefaultAnimationScheduler::ConstructL()
1.213 + {
1.214 + _LIT(KOnInactive,"ONINACTIVE");
1.215 + _LIT(KStopAnimation,"STOPANIMATION");
1.216 + _LIT(KStopAllDrawing,"STOPALLDRAWING");
1.217 + _LIT(KIgnore,"IGNORE");
1.218 +
1.219 + TPtrC inactivityBehaviourString;
1.220 + WsIniFile->FindVar(KOnInactive,inactivityBehaviourString);
1.221 + if(inactivityBehaviourString.CompareF(KStopAnimation)==0)
1.222 + iInactivityBehaviour = EStopAnimation;
1.223 + else if(inactivityBehaviourString.CompareF(KStopAllDrawing)==0)
1.224 + iInactivityBehaviour = EStopAllDrawing;
1.225 + else if(inactivityBehaviourString.CompareF(KIgnore)==0)
1.226 + iInactivityBehaviour = EIgnore;
1.227 +
1.228 + _LIT(KRedrawGracePeriod, "REDRAWGRACEPERIOD");
1.229 + TInt tmp = KRedrawGrace;
1.230 + WsIniFile->FindVar(KRedrawGracePeriod, tmp);
1.231 + iRedrawGracePeriod = tmp;
1.232 +
1.233 + _LIT(KAnimationGracePeriod, "ANIMATIONGRACEPERIOD");
1.234 + tmp = KAnimationGrace;
1.235 + WsIniFile->FindVar(KAnimationGracePeriod, tmp);
1.236 + iAnimationGracePeriod = tmp;
1.237 +
1.238 + iIdleInitiator = new(ELeave) CAsyncCallBack(TCallBack(OnIdleCallBack,this),EWsGraphicAnimateAwaitIdlePriority);
1.239 +
1.240 + TInt screenCount;
1.241 + User::LeaveIfError(HAL::Get( HAL::EDisplayNumberOfScreens, screenCount));
1.242 + for (TInt i = 0; i < screenCount; i++)
1.243 + {
1.244 + CScreenState* screenState = CScreenState::NewL (this, i);
1.245 + CleanupStack::PushL(screenState);
1.246 + iScreenState.AppendL (screenState);
1.247 + CleanupStack::Pop(screenState);
1.248 + }
1.249 +
1.250 +
1.251 + // Ensure that the wait loop array has some allocated slots, making it highly
1.252 + // unlikely that Append() will fail due to OOM.
1.253 + iRedrawWaitLoop.ReserveL (8);
1.254 + }
1.255 +
1.256 +void CWindowServer::CDefaultAnimationScheduler::Invalidate(const TGraphicDrawerId& aId)
1.257 + {
1.258 + const TInt screenCount = iEnv.ScreenCount();
1.259 + for(TInt ii = 0; ii < screenCount; ii++)
1.260 + {
1.261 + CScreenState* screenState = iScreenState[ii];
1.262 + if(!screenState->iInvalidateAll)
1.263 + {
1.264 + switch(screenState->iInvalidated.InsertInOrder(aId,TLinearOrder<TGraphicDrawerId>(TGraphicDrawerId::Compare)))
1.265 + {
1.266 + case KErrNone:
1.267 + case KErrAlreadyExists:
1.268 + break;
1.269 + default:
1.270 + screenState->iInvalidateAll = ETrue;
1.271 + screenState->iInvalidated.Reset();
1.272 + }
1.273 + }
1.274 + }
1.275 + iIdleInitiator->CallBack();
1.276 + }
1.277 +
1.278 +CWindowServer::CDefaultAnimationScheduler::CScreenState::CScreenState (CDefaultAnimationScheduler* aScheduler, TInt aScreenNumber):
1.279 + CActive(EComposeCompletePriority)
1.280 + {
1.281 + iScreenUpdateDetails.iScheduler = aScheduler;
1.282 + iScreenUpdateDetails.iScreenNumber = aScreenNumber;
1.283 + }
1.284 +
1.285 +CWindowServer::CDefaultAnimationScheduler::CScreenState*
1.286 +CWindowServer::CDefaultAnimationScheduler::CScreenState::NewL (CDefaultAnimationScheduler* aScheduler, TInt aScreenOrdinal)
1.287 + {
1.288 + CScreenState* self = new(ELeave)CScreenState(aScheduler, aScreenOrdinal);
1.289 + CleanupStack::PushL(self);
1.290 + self->ConstructL();
1.291 + CleanupStack::Pop(self);
1.292 + return self;
1.293 + }
1.294 +
1.295 +void CWindowServer::CDefaultAnimationScheduler::CScreenState::ConstructL ()
1.296 + {
1.297 + iUpdateOn = CPeriodic::NewL(EComposeCompletePriority);
1.298 + iWaitingClients.ReserveL(8);
1.299 + CActiveScheduler::Add(this);
1.300 + }
1.301 +
1.302 +CWindowServer::CDefaultAnimationScheduler::CScreenState::~CScreenState()
1.303 + {
1.304 + CActive::Cancel();
1.305 + iInvalidated.Close();
1.306 + delete iUpdateOn;
1.307 + TInt i = iWaitingClients.Count();
1.308 + while(i--)
1.309 + {
1.310 + iWaitingClients[i].iObserver.ScreenUpdateComplete(KErrAbort);
1.311 + }
1.312 + iWaitingClients.Close();
1.313 + }
1.314 +
1.315 +void CWindowServer::CDefaultAnimationScheduler::CScreenState::SetActive()
1.316 + {
1.317 + CActive::SetActive ();
1.318 + }
1.319 +
1.320 +/**
1.321 +This function is called from CWsClient d'tor to make sure we will not hang on to any deleted objects.
1.322 +*/
1.323 +void CWindowServer::CDefaultAnimationScheduler::CScreenState::ClearScreenUpdateObserver(const MWsAnimationScheduler::MScreenUpdateObserver& aObserver)
1.324 + {
1.325 + const TInt count = iWaitingClients.Count();
1.326 + for(TInt i = count-1 ; i >= 0; i--)
1.327 + {
1.328 + if( &aObserver == &(iWaitingClients[i].iObserver) )
1.329 + {
1.330 + TWaitingClient& client = iWaitingClients[i];
1.331 + client.iObserver.ScreenUpdateComplete(KErrCancel);
1.332 + iWaitingClients.Remove(i);
1.333 + }
1.334 + }
1.335 + }
1.336 +
1.337 +void CWindowServer::CDefaultAnimationScheduler::CScreenState::WaitForRedraws(MWsAnimationScheduler::MScreenUpdateObserver& aObserver, TInt aNumRedraws)
1.338 + {
1.339 + const TUint targetFrame = iFrameCount + aNumRedraws;
1.340 + TWaitingClient request(aObserver, targetFrame);
1.341 + TInt err = iWaitingClients.Append(request);
1.342 + if(err != KErrNone)
1.343 + {
1.344 + //If OOM and already have 8 waiting clients we will not accept a 9th client
1.345 + aObserver.ScreenUpdateComplete(KErrNoMemory);
1.346 + }
1.347 + }
1.348 +
1.349 +void CWindowServer::CDefaultAnimationScheduler::CScreenState::ReleaseRemainingClients()
1.350 + {
1.351 + const TInt count = iWaitingClients.Count();
1.352 + for(TInt i = count-1; i >= 0; i--)
1.353 + {
1.354 + TWaitingClient& client = iWaitingClients[i];
1.355 + client.iObserver.ScreenUpdateComplete(KErrNone);
1.356 + iWaitingClients.Remove(i);
1.357 + }
1.358 + }
1.359 +
1.360 +void CWindowServer::CDefaultAnimationScheduler::CScreenState::ReleaseClientsWaitingFor(TUint aCurrentFrame)
1.361 + {
1.362 + const TInt count = iWaitingClients.Count();
1.363 + for(TInt i = count-1; i >= 0; i--)
1.364 + {
1.365 + TWaitingClient& client = iWaitingClients[i];
1.366 + if(aCurrentFrame == client.iTargetFrame)
1.367 + {
1.368 + client.iObserver.ScreenUpdateComplete(KErrNone);
1.369 + iWaitingClients.Remove(i);
1.370 + }
1.371 + }
1.372 + }
1.373 +
1.374 +/**
1.375 +Invoked when the rendering pipline signals that it is ready to receive updates.
1.376 +*/
1.377 +void CWindowServer::CDefaultAnimationScheduler::CScreenState::RunL()
1.378 + {
1.379 + iFrameCount++;
1.380 +
1.381 + //Complete any clients waiting for this frame
1.382 + ReleaseClientsWaitingFor(iFrameCount);
1.383 +
1.384 + iScreenUpdateDetails.iScheduler->ProcessUpdateCompletion (iScreenUpdateDetails.iScreenNumber);
1.385 +
1.386 + if(!IsActive())
1.387 + {
1.388 + //No further pending frames, release all remaining clients
1.389 + ReleaseRemainingClients();
1.390 + }
1.391 +
1.392 + }
1.393 +
1.394 +void CWindowServer::CDefaultAnimationScheduler::ProcessUpdateCompletion (TInt aScreenNumber)
1.395 + {
1.396 + // Stop all waitloops related to the specified screen.
1.397 + TInt waitLoopCount = iRedrawWaitLoop.Count();
1.398 + for (TInt waitLoop = 0; waitLoop < waitLoopCount; waitLoop++)
1.399 + if (iRedrawWaitLoop[waitLoop]->IsStarted() && (iRedrawWaitLoop[waitLoop]->iScreenNumber == aScreenNumber))
1.400 + iRedrawWaitLoop[waitLoop]->AsyncStop();
1.401 +
1.402 + // Perform any outstanding redraws on the specified screen.
1.403 + ScheduleUpdate (aScreenNumber, ETrue);
1.404 + }
1.405 +
1.406 +void CWindowServer::CDefaultAnimationScheduler::DoRedrawNow(MWsScreen& aScreen, MWsAnimationScheduler::MScreenUpdateObserver& aObserver)
1.407 + {
1.408 + TInt screenNumber = ScreenNumber (aScreen);
1.409 + TInt redrawCount = 0;
1.410 +
1.411 + // redrawCount is the number of times we should wait for redraws to complete.
1.412 + // If a redraw is not currently active then we need to wait (at most) once: for
1.413 + // any outstanding scheduled update to complete.
1.414 + // If a redraw is currently active then we need to wait (at most) twice: once for
1.415 + // the current update to complete, and once for any outstanding scheduled update to complete.
1.416 + if (!iScreenState[screenNumber]->IsActive())
1.417 + {
1.418 + // No animation in progress, so force a redraw of due updates.
1.419 + ScheduleUpdate(screenNumber, ETrue);
1.420 +
1.421 + // If there is still nothing drawing, set redrawCount to zero to make sure we do not wait.
1.422 + if (!iScreenState[screenNumber]->IsActive())
1.423 + {
1.424 + redrawCount = 0;
1.425 + aObserver.ScreenUpdateComplete(KErrNone);
1.426 + }
1.427 + else
1.428 + {
1.429 + redrawCount = 1;
1.430 + iScreenState[screenNumber]->WaitForRedraws(aObserver, redrawCount);
1.431 + }
1.432 + }
1.433 + else
1.434 + {
1.435 + redrawCount = 2;
1.436 + iScreenState[screenNumber]->WaitForRedraws(aObserver, redrawCount);
1.437 + }
1.438 + }
1.439 +
1.440 +void CWindowServer::CDefaultAnimationScheduler::ClearScreenUpdateObserver(const MWsAnimationScheduler::MScreenUpdateObserver& aObserver)
1.441 + {
1.442 + const TInt count = iScreenState.Count();
1.443 + for(TInt screenNumber=0; screenNumber<count; screenNumber++)
1.444 + {
1.445 + iScreenState[screenNumber]->ClearScreenUpdateObserver(aObserver);
1.446 + }
1.447 + }
1.448 +
1.449 +/**
1.450 +Switch to deactivate animation or drawing (based on setting of iInactivityBehaviour).
1.451 +See InvokeDueAnimation().
1.452 +*/
1.453 +void CWindowServer::CDefaultAnimationScheduler::OnInactive()
1.454 + {
1.455 + iInactive = ETrue;
1.456 + }
1.457 +
1.458 +/**
1.459 +Switch to deactivate/activate animation or drawing (based on setting of iInactivityBehaviour).
1.460 +See InvokeDueAnimation().
1.461 +*/
1.462 +void CWindowServer::CDefaultAnimationScheduler::OnActive()
1.463 + {
1.464 + iInactive = EFalse;
1.465 + if(iInactiveDraws)
1.466 + {
1.467 + iInactiveDraws = EFalse;
1.468 + iIdleInitiator->CallBack();
1.469 + }
1.470 + }
1.471 +
1.472 +void CWindowServer::CDefaultAnimationScheduler::ScheduleRedraw(MWsScreen& aScreen,const TTime& aWhen)
1.473 + {
1.474 + iRedrawScheduled = ETrue;
1.475 + ScheduleAnimation(aScreen, aWhen);
1.476 + }
1.477 +
1.478 +/**
1.479 +Given a MWsScreen pointer, return an integer value representing the ordinal position
1.480 +of the screen in the Window Server.
1.481 +*/
1.482 +TInt CWindowServer::CDefaultAnimationScheduler::ScreenNumber(MWsScreen& aScreen) const
1.483 + {
1.484 + TInt numberOfScreens = CWsTop::NumberOfScreens();
1.485 + TInt theScreen;
1.486 +
1.487 + for (theScreen = 0; theScreen < numberOfScreens; theScreen++)
1.488 + if (CWsTop::Screen(theScreen) == &aScreen)
1.489 + break;
1.490 +
1.491 + WS_ASSERT_DEBUG(theScreen < numberOfScreens, EWsPanicWsGraphic);
1.492 + return theScreen;
1.493 + }
1.494 +
1.495 +/**
1.496 +Perform redraw and return only when completed.
1.497 +NOTE: This method uses CActiveSchedulerWait to run a "modal wait loop" while the
1.498 + redraw complete signal is pending. When the signal is received, AsyncStop() is
1.499 + invoked on all active wait loops for the signalling screen.
1.500 +*/
1.501 +void CWindowServer::CDefaultAnimationScheduler::DoRedrawNow(MWsScreen& aScreen)
1.502 + {
1.503 + TInt screenNumber = ScreenNumber (aScreen);
1.504 + TInt redrawCount = 0;
1.505 +
1.506 + // redrawCount is the number of times we should wait for redraws to complete.
1.507 + // If a redraw is not currently active then we need to wait (at most) once: for
1.508 + // any outstanding scheduled update to complete.
1.509 + // If a redraw is currently active then we need to wait (at most) twice: once for
1.510 + // the current update to complete, and once for any outstanding scheduled update to complete.
1.511 + if (!iScreenState[screenNumber]->IsActive())
1.512 + {
1.513 + // No animation in progress, so force a redraw of due updates.
1.514 + ScheduleUpdate(screenNumber, ETrue);
1.515 +
1.516 + // If there is still nothing drawing, set redrawCount to zero to make sure we do not wait.
1.517 + if (!iScreenState[screenNumber]->IsActive())
1.518 + redrawCount = 0;
1.519 + else
1.520 + redrawCount = 1;
1.521 + }
1.522 + else
1.523 + redrawCount = 2;
1.524 +
1.525 + // Wait for the sepecified number of redraws.
1.526 + if (redrawCount)
1.527 + {
1.528 + // Allocate the wait loop on the stack so we are not subject to heap OOM.
1.529 + TBuf8<sizeof (CScreenUpdateWait)> buf;
1.530 + Mem::FillZ(&buf, sizeof (CScreenUpdateWait));
1.531 + CScreenUpdateWait* waitLoop = new (&buf) CScreenUpdateWait(screenNumber);
1.532 + if (iRedrawWaitLoop.Append(waitLoop) == KErrNone)
1.533 + {
1.534 + // Run the active scheduler while updates are active
1.535 + while (redrawCount-- && iScreenState[screenNumber]->IsActive())
1.536 + waitLoop->Start();
1.537 +
1.538 + iRedrawWaitLoop.Remove(iRedrawWaitLoop.Count() - 1);
1.539 + }
1.540 + waitLoop->~CScreenUpdateWait();
1.541 + }
1.542 + }
1.543 +
1.544 +/**
1.545 +Schedule an update for a specific screen at a given point in time.
1.546 +*/
1.547 +void CWindowServer::CDefaultAnimationScheduler::ScheduleAnimation(MWsScreen& aScreen, const TTime& aWhen)
1.548 + {
1.549 + TSchedule schedule;
1.550 + schedule.iScreen = &aScreen;
1.551 + schedule.iScheduled = ETrue;
1.552 + schedule.iWhen = aWhen;
1.553 + schedule.iScreenNumber = ScreenNumber (aScreen);
1.554 + schedule.iRedraw = iRedrawScheduled;
1.555 + iRedrawScheduled = EFalse;
1.556 + TBool ok = EFalse;
1.557 + const TInt idx = iSchedule.FindInUnsignedKeyOrder(schedule);
1.558 + if(0 <= idx)
1.559 + {
1.560 + TSchedule& currSchedule=iSchedule[idx];
1.561 + if(currSchedule.iScheduled)
1.562 + {
1.563 + if(currSchedule.iWhen > aWhen)
1.564 + {
1.565 + currSchedule.iWhen = aWhen;
1.566 + }
1.567 + }
1.568 + else
1.569 + {
1.570 + currSchedule = schedule;
1.571 + }
1.572 + ok = ETrue;
1.573 + }
1.574 + else
1.575 + ok = (KErrNone == iSchedule.InsertInUnsignedKeyOrder(schedule));
1.576 +
1.577 + if(ok)
1.578 + iIdleInitiator->CallBack();
1.579 + }
1.580 +
1.581 +void CWindowServer::CDefaultAnimationScheduler::UnscheduleAnimation(MWsScreen& aScreen)
1.582 + {
1.583 + TSchedule schedule;
1.584 + schedule.iScreen = &aScreen;
1.585 + const TInt idx = iSchedule.FindInUnsignedKeyOrder(schedule);
1.586 + if(0 <= idx)
1.587 + iSchedule[idx].iScheduled = EFalse;
1.588 + }
1.589 +
1.590 +TBool CWindowServer::CDefaultAnimationScheduler::OnIdleCallBack(TAny* aAny)
1.591 + {
1.592 + WS_ASSERT_DEBUG(aAny, EWsPanicWsGraphic);
1.593 +
1.594 + if(aAny)
1.595 + static_cast<CDefaultAnimationScheduler*>(aAny)->OnIdleCallBack(EFalse);
1.596 +
1.597 + return EFalse; //ignored by caller
1.598 + }
1.599 +
1.600 +void CWindowServer::CDefaultAnimationScheduler::OnIdleCallBack(TBool aForce)
1.601 + {
1.602 + const TInt screenCount = iEnv.ScreenCount();
1.603 + for(TInt ii = 0; ii < screenCount; ii++)
1.604 + ScheduleUpdate (ii, aForce);
1.605 + }
1.606 +
1.607 +
1.608 +
1.609 +/**
1.610 +@return The number of microseconds (from now) that the specified scheduled update should be run at. This
1.611 +takes into account any set grace period and protects the scheduler from entering an infinite loop servicing
1.612 +animations with back-to-back frame updates.
1.613 +*/
1.614 +TTimeIntervalMicroSeconds CWindowServer::CDefaultAnimationScheduler::GetDueDelta (TBool aForceRedraw, TSchedule* aScheduledUpdate)
1.615 + {
1.616 + WS_ASSERT_DEBUG(aScheduledUpdate, EWsPanicWsGraphic);
1.617 + WS_ASSERT_DEBUG(aScheduledUpdate->iScheduled, EWsPanicWsGraphic);
1.618 +
1.619 + TTime now;
1.620 + TInt64 grace = I64LIT(0);
1.621 + TTimeIntervalMicroSeconds thisUpdateDueIn = I64LIT(0); //Microseconds from now
1.622 +
1.623 + // Only use grace periods if not forcing due updates.
1.624 + if (!aForceRedraw)
1.625 + {
1.626 + if (aScheduledUpdate->iRedraw)
1.627 + grace = iRedrawGracePeriod;
1.628 + else
1.629 + grace = iAnimationGracePeriod;
1.630 + }
1.631 +
1.632 + now.UniversalTime();
1.633 + thisUpdateDueIn = aScheduledUpdate->iWhen.MicroSecondsFrom(now);
1.634 +
1.635 + // Add the grace period if the update is due in less time than the grace period.
1.636 + if (thisUpdateDueIn < grace)
1.637 + thisUpdateDueIn = grace;
1.638 + else if (thisUpdateDueIn > KHalfHour)
1.639 + thisUpdateDueIn = KHalfHour;
1.640 +
1.641 + return thisUpdateDueIn;
1.642 + }
1.643 +
1.644 +/**
1.645 +Schedule an actual screen update at the point in time at which it is due. The due time may be modified by
1.646 +this method based on any "grace period" values.
1.647 +
1.648 +@param aScreen Screen number to update.
1.649 +@param aForceRedraw Force redraws that are due. This causes grace periods not to be used.
1.650 +*/
1.651 +void CWindowServer::CDefaultAnimationScheduler::ScheduleUpdate (TInt aScreenNumber, TBool aForceRedraw)
1.652 + {
1.653 + // Schedule updates for any invalidated regions.
1.654 + RedrawAllInvalidatedRegions (aScreenNumber);
1.655 +
1.656 + TSchedule* scheduledUpdate = GetScheduledScreenUpdate(aScreenNumber);
1.657 + if (scheduledUpdate)
1.658 + {
1.659 + WS_ASSERT_DEBUG(scheduledUpdate->iScheduled, EWsPanicWsGraphic);
1.660 + WS_ASSERT_DEBUG(aScreenNumber < iScreenState.Count(), EWsPanicWsGraphic);
1.661 +
1.662 + CScreenState& screenState = *iScreenState[aScreenNumber];
1.663 +
1.664 + // Initiate redraw if scheduled and not currently updating the display.
1.665 + if(!screenState.IsActive())
1.666 + {
1.667 + TTimeIntervalMicroSeconds thisUpdateDueIn =
1.668 + GetDueDelta (aForceRedraw, scheduledUpdate);
1.669 +
1.670 + // Reschedule any preexisting update if this one is due earlier.
1.671 + // If this update is not due earlier than a preexisting update then
1.672 + // there is nothing to do - just let the pending update occur.
1.673 + TTime now;
1.674 + now.UniversalTime();
1.675 + TBool performUpdate = ETrue;
1.676 + if(screenState.iUpdateOn->IsActive())
1.677 + {
1.678 + if (thisUpdateDueIn < screenState.iExpectedTickTime.MicroSecondsFrom(now))
1.679 + screenState.iUpdateOn->Cancel();
1.680 + else
1.681 + performUpdate = EFalse;
1.682 + }
1.683 +
1.684 + if (performUpdate)
1.685 + {
1.686 + if (thisUpdateDueIn.Int64() == 0) // Perform an immediate update if we are due.
1.687 + {
1.688 + screenState.iExpectedTickTime = now;
1.689 + InvokeDueAnimation(aScreenNumber);
1.690 + }
1.691 + else // Schedule the tick at the appropriate time.
1.692 + {
1.693 + WS_ASSERT_DEBUG(thisUpdateDueIn.Int64() > 0, EWsPanicWsGraphic);
1.694 + screenState.iExpectedTickTime = now + thisUpdateDueIn;
1.695 + screenState.iUpdateOn->Start(thisUpdateDueIn.Int64(),0,TCallBack(InvokeDueAnimation, &screenState.iScreenUpdateDetails));
1.696 + }
1.697 + }
1.698 + }
1.699 + }
1.700 + }
1.701 +
1.702 +/**
1.703 +@return A pointer to the scheduled update details currently associated with the specified screen.
1.704 + If there is no scheduled update then NULL is returned.
1.705 +@note There is only ever one scheduled update per screen.
1.706 +*/
1.707 +CWindowServer::CDefaultAnimationScheduler::TSchedule* CWindowServer::CDefaultAnimationScheduler::GetScheduledScreenUpdate(TInt aScreenNumber)
1.708 + {
1.709 + TSchedule* result = NULL;
1.710 + const TInt count = iSchedule.Count();
1.711 + for(TInt ii = 0; ii < count; ii++)
1.712 + {
1.713 + if (iSchedule[ii].iScreenNumber == aScreenNumber)
1.714 + {
1.715 + if (iSchedule[ii].iScheduled)
1.716 + result = &iSchedule[ii];
1.717 + break;
1.718 + }
1.719 + }
1.720 +
1.721 + return result;
1.722 + }
1.723 +
1.724 +/**
1.725 +Redraw invalidated graphic IDs. If invalid regions exist, this will cause ScheduleRedraw() to be invoked.
1.726 +*/
1.727 +void CWindowServer::CDefaultAnimationScheduler::RedrawAllInvalidatedRegions (TInt aScreen)
1.728 + {
1.729 + WS_ASSERT_DEBUG(iScreenState.Count() > aScreen, EWsPanicWsGraphic);
1.730 +
1.731 + CScreenState& screenState = *iScreenState[aScreen];
1.732 + if(screenState.iInvalidateAll || screenState.iInvalidated.Count())
1.733 + {
1.734 + const TArray<TGraphicDrawerId> invalidArray = screenState.iInvalidated.Array();
1.735 + MWsScreen* screen = iEnv.Screen(aScreen);
1.736 + WS_ASSERT_DEBUG(screen, EWsPanicWsGraphic);
1.737 + if(screen)
1.738 + {
1.739 + if(screenState.iInvalidateAll)
1.740 + Redraw(*screen);
1.741 + else
1.742 + RedrawInvalid(*screen, screenState.iInvalidated.Array());
1.743 + }
1.744 + screenState.iInvalidateAll = EFalse;
1.745 + }
1.746 + screenState.iInvalidated.Reset();
1.747 + }
1.748 +
1.749 +TBool CWindowServer::CDefaultAnimationScheduler::InvokeDueAnimation(TAny* aAny)
1.750 + {
1.751 + WS_ASSERT_DEBUG(aAny, EWsPanicWsGraphic);
1.752 + TScreenUpdateDetails* args = reinterpret_cast<TScreenUpdateDetails*>(aAny);
1.753 + if(args)
1.754 + args->iScheduler->InvokeDueAnimation (args->iScreenNumber);
1.755 +
1.756 + return EFalse;
1.757 + }
1.758 +
1.759 +void CWindowServer::CDefaultAnimationScheduler::InvokeDueAnimation(TInt aScreen)
1.760 + {
1.761 + WS_ASSERT_DEBUG(aScreen < iScreenState.Count(), EWsPanicWsGraphic);
1.762 + CScreenState& screenState = *iScreenState[aScreen];
1.763 + WS_ASSERT_DEBUG(!screenState.IsActive(), EWsPanicWsGraphic);
1.764 +
1.765 + // All updates are driven through ScheduleRedraw() and ScheduleAnimation().
1.766 + screenState.iUpdateOn->Cancel();
1.767 +
1.768 + TSchedule* scheduledUpdate = GetScheduledScreenUpdate(aScreen);
1.769 + if (scheduledUpdate)
1.770 + {
1.771 + WS_ASSERT_DEBUG(scheduledUpdate->iScheduled, EWsPanicWsGraphic);
1.772 +
1.773 + // Honour any flags that indicate we should not redraw.
1.774 + switch(iInactivityBehaviour)
1.775 + {
1.776 + case EStopAnimation :
1.777 + // Stop server side drawing. Only the client may redraw if iInactive is set.
1.778 + if(iInactive && !scheduledUpdate->iRedraw)
1.779 + {
1.780 + iInactiveDraws = ETrue;
1.781 + return;
1.782 + }
1.783 + break;
1.784 + case EStopAllDrawing :
1.785 + // Stop both client and server side drawing.
1.786 + if(iInactive)
1.787 + {
1.788 + iInactiveDraws = ETrue;
1.789 + return;
1.790 + }
1.791 + break;
1.792 + case EIgnore :
1.793 + break;
1.794 + default :
1.795 + WS_ASSERT_DEBUG(EFalse, EWsPanicWsGraphic);
1.796 + break;
1.797 + }
1.798 +
1.799 + scheduledUpdate->iScheduled = EFalse;
1.800 + screenState.SetActive();
1.801 + Animate(*scheduledUpdate->iScreen, &(screenState.iStatus));
1.802 + }
1.803 + }
1.804 +
1.805 +// CWindowServer::CServer \\\\\\\\\\\\\\\\\\\\\\
1.806 +
1.807 +class CWindowServer::CServer : public CPolicyServer
1.808 + {
1.809 +public:
1.810 + static CServer* NewL()
1.811 + {
1.812 + return new(ELeave) CServer;
1.813 + }
1.814 + void StartL()
1.815 + {
1.816 + CPolicyServer::StartL(KWSERVServerName);
1.817 + }
1.818 + TInt SessionCount()
1.819 + {
1.820 + iSessionIter.SetToFirst();
1.821 + TInt count=0;
1.822 + while(iSessionIter++)
1.823 + ++count;
1.824 + return(count);
1.825 + }
1.826 +
1.827 +public: //from CPolicyServer
1.828 + /** Creates a new client for this server. */
1.829 + CSession2* NewSessionL(const TVersion& aVersion, const RMessage2& aMessage) const
1.830 + {
1.831 + TVersion v(KWservMajorVersionNumber, KWservMinorVersionNumber, KWservBuildVersionNumber);
1.832 + if (User::QueryVersionSupported(v, aVersion)==EFalse)
1.833 + User::Leave(KErrNotSupported);
1.834 + RThread thread;
1.835 + User::LeaveIfError(aMessage.Client(thread));
1.836 + return(new(ELeave) CWsClient(thread));
1.837 + }
1.838 +private:
1.839 + CServer() : CPolicyServer(EMainServerPriority, KWsServPolicy)
1.840 + {}
1.841 + };
1.842 +
1.843 +
1.844 +// CWindowServer \\\\\\\\\\\\\\\\\\\\\\\\\\\
1.845 +
1.846 +CWindowServer *CWindowServer::NewL()
1.847 +//
1.848 +// Create a new CWindowServer.
1.849 +//
1.850 + {
1.851 + CWindowServer* self = new(ELeave) CWindowServer();
1.852 + CleanupStack::PushL(self);
1.853 + self->ConstructL();
1.854 + CleanupStack::Pop(self);
1.855 + return self;
1.856 + }
1.857 +
1.858 +CWindowServer::CWindowServer()
1.859 +//
1.860 +// Constructor.
1.861 +//
1.862 + {
1.863 + }
1.864 +
1.865 +CWindowServer::~CWindowServer()
1.866 + {
1.867 + delete iServer;
1.868 +
1.869 + iMemoryReleases.Reset();
1.870 + WS_ASSERT_DEBUG(iDrawerMasterIndex.IsEmpty(), EWsPanicWsGraphic);
1.871 + iDrawerMasterIndex.Close();
1.872 +
1.873 + delete iDefaultAnimationScheduler;
1.874 + iDefaultAnimationScheduler = NULL; //might be called from clients during server destruction
1.875 + }
1.876 +
1.877 +void CWindowServer::ConstructL()
1.878 + {
1.879 + iServer = CServer::NewL();
1.880 + CWsTop::PluginManager()->InitializePluginsL(*this); // plugins are loaded and own by CWsTop
1.881 + iDefaultAnimationScheduler = new(ELeave) CDefaultAnimationScheduler(*this);
1.882 + iDefaultAnimationScheduler->ConstructL();
1.883 + RegisterMemoryRelease(this);
1.884 + }
1.885 +
1.886 +void CWindowServer::StartL()
1.887 + {
1.888 + iServer->StartL();
1.889 + }
1.890 +
1.891 +void CWindowServer::SetPinClientDescriptors(TBool aPin)
1.892 + {
1.893 + iServer->SetPinClientDescriptors(aPin);
1.894 + }
1.895 +
1.896 +TInt CWindowServer::SessionCount()
1.897 + {
1.898 + return iServer->SessionCount();
1.899 + }
1.900 +
1.901 +const CWsGraphicDrawer* CWindowServer::ResolveGraphic(const TGraphicDrawerId& aId) const
1.902 + {
1.903 + return iDrawerMasterIndex.ResolveGraphic(aId);
1.904 + }
1.905 +
1.906 +void CWindowServer::Invalidate(const TGraphicDrawerId& aId)
1.907 + {
1.908 + AnimationScheduler()->Invalidate(aId);
1.909 + }
1.910 +
1.911 +TInt CWindowServer::ScreenCount() const
1.912 + {
1.913 + return CWsTop::NumberOfScreens();
1.914 + }
1.915 +
1.916 +MWsScreen* CWindowServer::Screen(TInt aIndex)
1.917 + {
1.918 + if((aIndex >= 0) && (aIndex < ScreenCount()))
1.919 + {
1.920 + return CWsTop::Screen(aIndex);
1.921 + }
1.922 + return NULL;
1.923 + }
1.924 +
1.925 +const MWsScreen* CWindowServer::Screen(TInt aIndex) const
1.926 + {
1.927 + if((aIndex >= 0) && (aIndex < ScreenCount()))
1.928 + {
1.929 + return CWsTop::Screen(aIndex);
1.930 + }
1.931 + return NULL;
1.932 + }
1.933 +
1.934 +/**
1.935 +Custom Animation Scheduler
1.936 +*/
1.937 +TBool CWindowServer::SetCustomAnimationScheduler(MWsAnimationScheduler* /*aScheduler*/)
1.938 + {
1.939 + return EFalse;
1.940 + }
1.941 +
1.942 +TBool CWindowServer::HasCustomAnimationScheduler() const
1.943 + {
1.944 + return EFalse;
1.945 + }
1.946 +
1.947 +TBool CWindowServer::ClearCustomAnimationScheduler(MWsAnimationScheduler* /*aCurrentScheduler*/)
1.948 + {
1.949 + return EFalse;
1.950 + }
1.951 +
1.952 +MWsAnimationScheduler* CWindowServer::AnimationScheduler()
1.953 + {
1.954 + return iDefaultAnimationScheduler;
1.955 + }
1.956 +
1.957 +void CWindowServer::PrepareShutdown()
1.958 + {
1.959 + //Stop the renderloop, i.e. prevent any further calls to MWsAnimationScheduler::Animate()
1.960 + delete iDefaultAnimationScheduler;
1.961 + iDefaultAnimationScheduler = NULL;
1.962 + }
1.963 +
1.964 +TInt CWindowServer::RegisterEventHandler(CWsGraphicDrawer* aDrawer, MWsEventHandler* aHandler, TUint32 aEventMask)
1.965 + {
1.966 + if (!aDrawer || !aHandler || aEventMask==0)
1.967 + return KErrArgument;
1.968 + TInt err = TWindowServerEvent::RegisterDrawerHandler(aDrawer, aEventMask);
1.969 + if (err != KErrNone)
1.970 + return err;
1.971 + aDrawer->SetEventHandler(aHandler);
1.972 + return KErrNone;
1.973 + }
1.974 +
1.975 +TInt CWindowServer::UnregisterEventHandler(CWsGraphicDrawer* aDrawer)
1.976 + {
1.977 + if (!aDrawer || (aDrawer && !aDrawer->HasEventHandler()))
1.978 + return KErrArgument;
1.979 + TInt err = TWindowServerEvent::UnregisterDrawerHandler(aDrawer);
1.980 + if (err != KErrNone)
1.981 + return err;
1.982 + aDrawer->SetEventHandler(NULL);
1.983 + return KErrNone;
1.984 + }
1.985 +
1.986 +TInt CWindowServer::RegisterWsEventHandler(MWsEventHandler* aHandler, TUint32 aEventMask)
1.987 + {
1.988 + if (!aHandler || aEventMask==0)
1.989 + return KErrArgument;
1.990 + return TWindowServerEvent::RegisterWsEventHandler(aHandler, aEventMask);
1.991 + }
1.992 +
1.993 +TInt CWindowServer::UnregisterWsEventHandler(MWsEventHandler* aHandler)
1.994 + {
1.995 + return TWindowServerEvent::UnregisterWsEventHandler(aHandler);
1.996 + }
1.997 +
1.998 +TAny* CWindowServer::ResolveObjectInterface(TUint aTypeId)
1.999 + {
1.1000 + switch(aTypeId)
1.1001 + {
1.1002 + case MWsActiveSchedulerDebug::EWsObjectInterfaceId:
1.1003 + return static_cast<MWsActiveSchedulerDebug*>(CWsActiveScheduler::Static());
1.1004 + case MWsIniFile::EWsObjectInterfaceId:
1.1005 + return static_cast<MWsIniFile*>(WsIniFile);
1.1006 + }
1.1007 +
1.1008 + if (CWsPluginManager *plugMgr=CWsTop::PluginManager())
1.1009 + return plugMgr->ResolveObjectInterface(aTypeId);
1.1010 +
1.1011 + return NULL;
1.1012 + }
1.1013 +
1.1014 +void CWindowServer::Log(TInt aPriority,const TDesC &aFmt,TInt aParam)
1.1015 + {
1.1016 + if (wsDebugLog)
1.1017 + {
1.1018 + wsDebugLog->MiscMessage(aPriority, aFmt, aParam);
1.1019 + }
1.1020 + }
1.1021 +
1.1022 +// CWsGraphicDrawer master index
1.1023 +
1.1024 +TInt CWindowServer::AddGraphicDrawer(CWsGraphicDrawer* aDrawer)
1.1025 + {
1.1026 + return iDrawerMasterIndex.Add(aDrawer);
1.1027 + }
1.1028 +
1.1029 +TInt CWindowServer::SwapGraphicDrawer(CWsGraphicDrawer* aDrawer)
1.1030 + {
1.1031 + return iDrawerMasterIndex.Swap(aDrawer);
1.1032 + }
1.1033 +
1.1034 +TInt CWindowServer::RemoveGraphicDrawer(const TGraphicDrawerId& aId)
1.1035 + {
1.1036 + return iDrawerMasterIndex.Remove(aId);
1.1037 + }
1.1038 +
1.1039 +TInt CWindowServer::RemoveAllGraphicDrawers(const MWsClient& aOwner)
1.1040 + {
1.1041 + return iDrawerMasterIndex.RemoveAll(aOwner);
1.1042 + }
1.1043 +
1.1044 +TInt CWindowServer::RegisterMemoryRelease(MWsMemoryRelease * aMemoryRelease)
1.1045 + {
1.1046 + return iMemoryReleases.Append(aMemoryRelease);
1.1047 + }
1.1048 +
1.1049 +void CWindowServer::UnregisterMemoryRelease(MWsMemoryRelease * aMemoryRelease)
1.1050 + {
1.1051 + for (TInt ii = iMemoryReleases.Count() - 1; ii >= 0; --ii)
1.1052 + {
1.1053 + if (iMemoryReleases[ii] == aMemoryRelease)
1.1054 + {
1.1055 + iMemoryReleases.Remove(ii);
1.1056 + break;
1.1057 + }
1.1058 + }
1.1059 + }
1.1060 +
1.1061 +TBool CWindowServer::ReleaseMemory(TMemoryReleaseLevel aLevel)
1.1062 + {
1.1063 + return CWsWindow::ReleaseMemory(aLevel);
1.1064 + }
1.1065 +
1.1066 +TBool CWindowServer::ReleaseMemory()
1.1067 + {
1.1068 + TBool released = EFalse;
1.1069 + for (TInt level = MWsMemoryRelease::ELow; !released && level <= MWsMemoryRelease::EHigh; ++level)
1.1070 + {
1.1071 + for (TInt ii = iMemoryReleases.Count() - 1; !released && ii >= 0; --ii)
1.1072 + {
1.1073 + released = iMemoryReleases[ii]->ReleaseMemory(static_cast<TMemoryReleaseLevel>(level));
1.1074 + }
1.1075 + }
1.1076 + return released;
1.1077 + }
1.1078 +
1.1079 +void CWindowServer::DestroySessionsForShutdown()
1.1080 + {
1.1081 + delete iServer;
1.1082 + iServer = NULL;
1.1083 + }
1.1084 +