First public contribution.
1 // Copyright (c) 1995-2009 Nokia Corporation and/or its subsidiary(-ies).
2 // All rights reserved.
3 // This component and the accompanying materials are made available
4 // under the terms of "Eclipse Public License v1.0"
5 // which accompanies this distribution, and is available
6 // at the URL "http://www.eclipse.org/legal/epl-v10.html".
8 // Initial Contributors:
9 // Nokia Corporation - initial contribution.
14 // Window server 'server' class
25 #include "wspluginmanager.h"
26 #include "graphics/windowserverconstants.h"
28 GLREF_D CDebugLogBase *wsDebugLog;
30 const TUint KRangeCount = 1;
31 // We use a lot of 64 bit time calculations, but a periodic can only cope with signed 32 bit times
32 // which gives them a limit of about 35 minutes.
33 // Fortunately, our animtions are safe if redrawn early. Every half an hour isn't going to hurt.
34 const TTimeIntervalMicroSeconds KHalfHour = 30 * 60 * 1000 * 1000;
36 const TInt KWsServRanges[KRangeCount] =
41 const TUint8 KElementsIndex[KRangeCount] =
43 CPolicyServer::EAlwaysPass,
46 const CPolicyServer::TPolicyElement KPolicyElements[] =
48 {_INIT_SECURITY_POLICY_C1(ECapabilityPowerMgmt), CPolicyServer::EFailClient},
49 {_INIT_SECURITY_POLICY_C1(ECapabilitySwEvent), CPolicyServer::EFailClient},
50 {_INIT_SECURITY_POLICY_C1(ECapabilityWriteDeviceData), CPolicyServer::EFailClient}
53 const CPolicyServer::TPolicy KWsServPolicy =
55 CPolicyServer::EAlwaysPass,
62 // CWindowServer::CDefaultAnimationScheduler \\\\\\\\\\\\\\\\\\\\\\\\\\\
64 class CWindowServer::CDefaultAnimationScheduler: public CBase, public MWsAnimationScheduler
66 // Associates a screen number with a CActiveSchedulerWait intance. This is used to
67 // achieve synchronous update completion on a specific screen.
68 class CScreenUpdateWait : public CActiveSchedulerWait
71 CScreenUpdateWait(TInt aScreenNumber) : iScreenNumber (aScreenNumber) {}
75 struct TScreenUpdateDetails
77 CWindowServer::CDefaultAnimationScheduler* iScheduler;
83 MWsScreen* iScreen; // used as a unique index, searching with FindInUnsignedKeyOrder
90 enum TInactivityBehaviour
98 CDefaultAnimationScheduler(MWsGraphicDrawerEnvironment& aEnv);
99 ~CDefaultAnimationScheduler();
100 void ConstructL(); //LeaveScan: member of nested class declaration
101 // implementing MWsAnimationScheduler
102 void ScheduleAnimation(MWsScreen& aScreen,const TTime& aWhen);
103 void UnscheduleAnimation(MWsScreen& aScreen);
104 void Invalidate(const TGraphicDrawerId& aId);
107 void ScheduleRedraw(MWsScreen& aScreen,const TTime& aWhen);
108 void DoRedrawNow(MWsScreen& aScreen);
109 void DoRedrawNow(MWsScreen& aScreen, MWsAnimationScheduler::MScreenUpdateObserver& aObserver);
110 void ClearScreenUpdateObserver(const MWsAnimationScheduler::MScreenUpdateObserver& aObserver);
112 static TBool OnIdleCallBack(TAny* aAny);
113 void ScheduleUpdate (TInt aScreenNumber, TBool aForce);
114 void OnIdleCallBack(TBool aForce);
115 static TBool InvokeDueAnimation(TAny* aAny);
116 void RedrawAllInvalidatedRegions (TInt aScreen);
117 TSchedule* GetScheduledScreenUpdate(TInt aScreen);
118 void InvokeDueAnimation(TInt aScreen);
119 void ProcessUpdateCompletion (TInt aScreenNumber);
120 TInt ScreenNumber(MWsScreen& aScreen) const;
121 TTimeIntervalMicroSeconds GetDueDelta (TBool aForceRedraw, TSchedule* aScheduledUpdate);
123 RPointerArray<CScreenUpdateWait> iRedrawWaitLoop;
124 RPointerArray<CScreenState> iScreenState;
125 MWsGraphicDrawerEnvironment& iEnv;
126 static const TInt64 KRedrawGrace;
127 static const TInt64 KAnimationGrace;
128 CAsyncCallBack* iIdleInitiator;
130 RArray<TSchedule> iSchedule;
132 TBool iInactiveDraws;
133 TBool iRedrawScheduled;
134 TInt64 iRedrawGracePeriod;
135 TInt64 iAnimationGracePeriod;
136 TInactivityBehaviour iInactivityBehaviour;
139 class CWindowServer::CDefaultAnimationScheduler::CScreenState : public CActive
142 static CScreenState* NewL (CDefaultAnimationScheduler* aScheduler, TInt aScreenOrdinal);
146 void WaitForRedraws(MWsAnimationScheduler::MScreenUpdateObserver& aObserver, TInt aNumRedraws);
147 void ClearScreenUpdateObserver(const MWsAnimationScheduler::MScreenUpdateObserver& aObserver);
148 CPeriodic* iUpdateOn;
149 TTime iExpectedTickTime;
150 TScreenUpdateDetails iScreenUpdateDetails;
151 RArray<TGraphicDrawerId> iInvalidated;
152 TBool iInvalidateAll;
155 CScreenState (CDefaultAnimationScheduler* aScheduler, TInt aScreenOrdinal);
157 void ReleaseRemainingClients();
158 void ReleaseClientsWaitingFor(TUint aCurrentFrame);
163 TRequestStatus* tmpTRS = &iStatus;
164 User::RequestComplete(tmpTRS, KErrNone);
169 TWaitingClient(MWsAnimationScheduler::MScreenUpdateObserver& aObserver, TInt aTargetFrame)
170 : iObserver(aObserver), iTargetFrame(aTargetFrame)
173 MWsAnimationScheduler::MScreenUpdateObserver& iObserver;
177 RArray<TWaitingClient> iWaitingClients;
180 // If using the default animation scheduler on a device, these two numbers may be worth tweaking in the inifile
181 // However, both are maximum periods - wserv will go faster than either if nothing else is using the system.
182 const TInt64 CWindowServer::CDefaultAnimationScheduler::KRedrawGrace = 0; // do redraws immediately
183 const TInt64 CWindowServer::CDefaultAnimationScheduler::KAnimationGrace = 0; // do animation redraws immediately
185 CWindowServer::CDefaultAnimationScheduler::CDefaultAnimationScheduler(MWsGraphicDrawerEnvironment& aEnv):
186 iEnv(aEnv), iSchedule(1,_FOFF(TSchedule,iScreen))
190 CWindowServer::CDefaultAnimationScheduler::~CDefaultAnimationScheduler()
193 delete iIdleInitiator;
195 TInt screenCount = iScreenState.Count();
196 for (TInt ii = 0; ii < screenCount; ii++)
197 delete iScreenState[ii];
199 iScreenState.Close();
201 // Destroy the redraw wait loops associated each screen, and close
202 // the associated Array objects.
203 TInt waitLoopCount = iRedrawWaitLoop.Count();
204 for (TInt waitLoop = 0; waitLoop < waitLoopCount; waitLoop++)
205 delete iRedrawWaitLoop[waitLoop];
206 iRedrawWaitLoop.Close();
209 void CWindowServer::CDefaultAnimationScheduler::ConstructL()
211 _LIT(KOnInactive,"ONINACTIVE");
212 _LIT(KStopAnimation,"STOPANIMATION");
213 _LIT(KStopAllDrawing,"STOPALLDRAWING");
214 _LIT(KIgnore,"IGNORE");
216 TPtrC inactivityBehaviourString;
217 WsIniFile->FindVar(KOnInactive,inactivityBehaviourString);
218 if(inactivityBehaviourString.CompareF(KStopAnimation)==0)
219 iInactivityBehaviour = EStopAnimation;
220 else if(inactivityBehaviourString.CompareF(KStopAllDrawing)==0)
221 iInactivityBehaviour = EStopAllDrawing;
222 else if(inactivityBehaviourString.CompareF(KIgnore)==0)
223 iInactivityBehaviour = EIgnore;
225 _LIT(KRedrawGracePeriod, "REDRAWGRACEPERIOD");
226 TInt tmp = KRedrawGrace;
227 WsIniFile->FindVar(KRedrawGracePeriod, tmp);
228 iRedrawGracePeriod = tmp;
230 _LIT(KAnimationGracePeriod, "ANIMATIONGRACEPERIOD");
231 tmp = KAnimationGrace;
232 WsIniFile->FindVar(KAnimationGracePeriod, tmp);
233 iAnimationGracePeriod = tmp;
235 iIdleInitiator = new(ELeave) CAsyncCallBack(TCallBack(OnIdleCallBack,this),EWsGraphicAnimateAwaitIdlePriority);
238 User::LeaveIfError(HAL::Get( HAL::EDisplayNumberOfScreens, screenCount));
239 for (TInt i = 0; i < screenCount; i++)
241 CScreenState* screenState = CScreenState::NewL (this, i);
242 CleanupStack::PushL(screenState);
243 iScreenState.AppendL (screenState);
244 CleanupStack::Pop(screenState);
248 // Ensure that the wait loop array has some allocated slots, making it highly
249 // unlikely that Append() will fail due to OOM.
250 iRedrawWaitLoop.ReserveL (8);
253 void CWindowServer::CDefaultAnimationScheduler::Invalidate(const TGraphicDrawerId& aId)
255 const TInt screenCount = iEnv.ScreenCount();
256 for(TInt ii = 0; ii < screenCount; ii++)
258 CScreenState* screenState = iScreenState[ii];
259 if(!screenState->iInvalidateAll)
261 switch(screenState->iInvalidated.InsertInOrder(aId,TLinearOrder<TGraphicDrawerId>(TGraphicDrawerId::Compare)))
264 case KErrAlreadyExists:
267 screenState->iInvalidateAll = ETrue;
268 screenState->iInvalidated.Reset();
272 iIdleInitiator->CallBack();
275 CWindowServer::CDefaultAnimationScheduler::CScreenState::CScreenState (CDefaultAnimationScheduler* aScheduler, TInt aScreenNumber):
276 CActive(EComposeCompletePriority)
278 iScreenUpdateDetails.iScheduler = aScheduler;
279 iScreenUpdateDetails.iScreenNumber = aScreenNumber;
282 CWindowServer::CDefaultAnimationScheduler::CScreenState*
283 CWindowServer::CDefaultAnimationScheduler::CScreenState::NewL (CDefaultAnimationScheduler* aScheduler, TInt aScreenOrdinal)
285 CScreenState* self = new(ELeave)CScreenState(aScheduler, aScreenOrdinal);
286 CleanupStack::PushL(self);
288 CleanupStack::Pop(self);
292 void CWindowServer::CDefaultAnimationScheduler::CScreenState::ConstructL ()
294 iUpdateOn = CPeriodic::NewL(EComposeCompletePriority);
295 iWaitingClients.ReserveL(8);
296 CActiveScheduler::Add(this);
299 CWindowServer::CDefaultAnimationScheduler::CScreenState::~CScreenState()
302 iInvalidated.Close();
304 TInt i = iWaitingClients.Count();
307 iWaitingClients[i].iObserver.ScreenUpdateComplete(KErrAbort);
309 iWaitingClients.Close();
312 void CWindowServer::CDefaultAnimationScheduler::CScreenState::SetActive()
314 CActive::SetActive ();
318 This function is called from CWsClient d'tor to make sure we will not hang on to any deleted objects.
320 void CWindowServer::CDefaultAnimationScheduler::CScreenState::ClearScreenUpdateObserver(const MWsAnimationScheduler::MScreenUpdateObserver& aObserver)
322 const TInt count = iWaitingClients.Count();
323 for(TInt i = count-1 ; i >= 0; i--)
325 if( &aObserver == &(iWaitingClients[i].iObserver) )
327 TWaitingClient& client = iWaitingClients[i];
328 client.iObserver.ScreenUpdateComplete(KErrCancel);
329 iWaitingClients.Remove(i);
334 void CWindowServer::CDefaultAnimationScheduler::CScreenState::WaitForRedraws(MWsAnimationScheduler::MScreenUpdateObserver& aObserver, TInt aNumRedraws)
336 const TUint targetFrame = iFrameCount + aNumRedraws;
337 TWaitingClient request(aObserver, targetFrame);
338 TInt err = iWaitingClients.Append(request);
341 //If OOM and already have 8 waiting clients we will not accept a 9th client
342 aObserver.ScreenUpdateComplete(KErrNoMemory);
346 void CWindowServer::CDefaultAnimationScheduler::CScreenState::ReleaseRemainingClients()
348 const TInt count = iWaitingClients.Count();
349 for(TInt i = count-1; i >= 0; i--)
351 TWaitingClient& client = iWaitingClients[i];
352 client.iObserver.ScreenUpdateComplete(KErrNone);
353 iWaitingClients.Remove(i);
357 void CWindowServer::CDefaultAnimationScheduler::CScreenState::ReleaseClientsWaitingFor(TUint aCurrentFrame)
359 const TInt count = iWaitingClients.Count();
360 for(TInt i = count-1; i >= 0; i--)
362 TWaitingClient& client = iWaitingClients[i];
363 if(aCurrentFrame == client.iTargetFrame)
365 client.iObserver.ScreenUpdateComplete(KErrNone);
366 iWaitingClients.Remove(i);
372 Invoked when the rendering pipline signals that it is ready to receive updates.
374 void CWindowServer::CDefaultAnimationScheduler::CScreenState::RunL()
378 //Complete any clients waiting for this frame
379 ReleaseClientsWaitingFor(iFrameCount);
381 iScreenUpdateDetails.iScheduler->ProcessUpdateCompletion (iScreenUpdateDetails.iScreenNumber);
385 //No further pending frames, release all remaining clients
386 ReleaseRemainingClients();
391 void CWindowServer::CDefaultAnimationScheduler::ProcessUpdateCompletion (TInt aScreenNumber)
393 // Stop all waitloops related to the specified screen.
394 TInt waitLoopCount = iRedrawWaitLoop.Count();
395 for (TInt waitLoop = 0; waitLoop < waitLoopCount; waitLoop++)
396 if (iRedrawWaitLoop[waitLoop]->IsStarted() && (iRedrawWaitLoop[waitLoop]->iScreenNumber == aScreenNumber))
397 iRedrawWaitLoop[waitLoop]->AsyncStop();
399 // Perform any outstanding redraws on the specified screen.
400 ScheduleUpdate (aScreenNumber, ETrue);
403 void CWindowServer::CDefaultAnimationScheduler::DoRedrawNow(MWsScreen& aScreen, MWsAnimationScheduler::MScreenUpdateObserver& aObserver)
405 TInt screenNumber = ScreenNumber (aScreen);
406 TInt redrawCount = 0;
408 // redrawCount is the number of times we should wait for redraws to complete.
409 // If a redraw is not currently active then we need to wait (at most) once: for
410 // any outstanding scheduled update to complete.
411 // If a redraw is currently active then we need to wait (at most) twice: once for
412 // the current update to complete, and once for any outstanding scheduled update to complete.
413 if (!iScreenState[screenNumber]->IsActive())
415 // No animation in progress, so force a redraw of due updates.
416 ScheduleUpdate(screenNumber, ETrue);
418 // If there is still nothing drawing, set redrawCount to zero to make sure we do not wait.
419 if (!iScreenState[screenNumber]->IsActive())
422 aObserver.ScreenUpdateComplete(KErrNone);
427 iScreenState[screenNumber]->WaitForRedraws(aObserver, redrawCount);
433 iScreenState[screenNumber]->WaitForRedraws(aObserver, redrawCount);
437 void CWindowServer::CDefaultAnimationScheduler::ClearScreenUpdateObserver(const MWsAnimationScheduler::MScreenUpdateObserver& aObserver)
439 const TInt count = iScreenState.Count();
440 for(TInt screenNumber=0; screenNumber<count; screenNumber++)
442 iScreenState[screenNumber]->ClearScreenUpdateObserver(aObserver);
447 Switch to deactivate animation or drawing (based on setting of iInactivityBehaviour).
448 See InvokeDueAnimation().
450 void CWindowServer::CDefaultAnimationScheduler::OnInactive()
456 Switch to deactivate/activate animation or drawing (based on setting of iInactivityBehaviour).
457 See InvokeDueAnimation().
459 void CWindowServer::CDefaultAnimationScheduler::OnActive()
464 iInactiveDraws = EFalse;
465 iIdleInitiator->CallBack();
469 void CWindowServer::CDefaultAnimationScheduler::ScheduleRedraw(MWsScreen& aScreen,const TTime& aWhen)
471 iRedrawScheduled = ETrue;
472 ScheduleAnimation(aScreen, aWhen);
476 Given a MWsScreen pointer, return an integer value representing the ordinal position
477 of the screen in the Window Server.
479 TInt CWindowServer::CDefaultAnimationScheduler::ScreenNumber(MWsScreen& aScreen) const
481 TInt numberOfScreens = CWsTop::NumberOfScreens();
484 for (theScreen = 0; theScreen < numberOfScreens; theScreen++)
485 if (CWsTop::Screen(theScreen) == &aScreen)
488 WS_ASSERT_DEBUG(theScreen < numberOfScreens, EWsPanicWsGraphic);
493 Perform redraw and return only when completed.
494 NOTE: This method uses CActiveSchedulerWait to run a "modal wait loop" while the
495 redraw complete signal is pending. When the signal is received, AsyncStop() is
496 invoked on all active wait loops for the signalling screen.
498 void CWindowServer::CDefaultAnimationScheduler::DoRedrawNow(MWsScreen& aScreen)
500 TInt screenNumber = ScreenNumber (aScreen);
501 TInt redrawCount = 0;
503 // redrawCount is the number of times we should wait for redraws to complete.
504 // If a redraw is not currently active then we need to wait (at most) once: for
505 // any outstanding scheduled update to complete.
506 // If a redraw is currently active then we need to wait (at most) twice: once for
507 // the current update to complete, and once for any outstanding scheduled update to complete.
508 if (!iScreenState[screenNumber]->IsActive())
510 // No animation in progress, so force a redraw of due updates.
511 ScheduleUpdate(screenNumber, ETrue);
513 // If there is still nothing drawing, set redrawCount to zero to make sure we do not wait.
514 if (!iScreenState[screenNumber]->IsActive())
522 // Wait for the sepecified number of redraws.
525 // Allocate the wait loop on the stack so we are not subject to heap OOM.
526 TBuf8<sizeof (CScreenUpdateWait)> buf;
527 Mem::FillZ(&buf, sizeof (CScreenUpdateWait));
528 CScreenUpdateWait* waitLoop = new (&buf) CScreenUpdateWait(screenNumber);
529 if (iRedrawWaitLoop.Append(waitLoop) == KErrNone)
531 // Run the active scheduler while updates are active
532 while (redrawCount-- && iScreenState[screenNumber]->IsActive())
535 iRedrawWaitLoop.Remove(iRedrawWaitLoop.Count() - 1);
537 waitLoop->~CScreenUpdateWait();
542 Schedule an update for a specific screen at a given point in time.
544 void CWindowServer::CDefaultAnimationScheduler::ScheduleAnimation(MWsScreen& aScreen, const TTime& aWhen)
547 schedule.iScreen = &aScreen;
548 schedule.iScheduled = ETrue;
549 schedule.iWhen = aWhen;
550 schedule.iScreenNumber = ScreenNumber (aScreen);
551 schedule.iRedraw = iRedrawScheduled;
552 iRedrawScheduled = EFalse;
554 const TInt idx = iSchedule.FindInUnsignedKeyOrder(schedule);
557 TSchedule& currSchedule=iSchedule[idx];
558 if(currSchedule.iScheduled)
560 if(currSchedule.iWhen > aWhen)
562 currSchedule.iWhen = aWhen;
567 currSchedule = schedule;
572 ok = (KErrNone == iSchedule.InsertInUnsignedKeyOrder(schedule));
575 iIdleInitiator->CallBack();
578 void CWindowServer::CDefaultAnimationScheduler::UnscheduleAnimation(MWsScreen& aScreen)
581 schedule.iScreen = &aScreen;
582 const TInt idx = iSchedule.FindInUnsignedKeyOrder(schedule);
584 iSchedule[idx].iScheduled = EFalse;
587 TBool CWindowServer::CDefaultAnimationScheduler::OnIdleCallBack(TAny* aAny)
589 WS_ASSERT_DEBUG(aAny, EWsPanicWsGraphic);
592 static_cast<CDefaultAnimationScheduler*>(aAny)->OnIdleCallBack(EFalse);
594 return EFalse; //ignored by caller
597 void CWindowServer::CDefaultAnimationScheduler::OnIdleCallBack(TBool aForce)
599 const TInt screenCount = iEnv.ScreenCount();
600 for(TInt ii = 0; ii < screenCount; ii++)
601 ScheduleUpdate (ii, aForce);
607 @return The number of microseconds (from now) that the specified scheduled update should be run at. This
608 takes into account any set grace period and protects the scheduler from entering an infinite loop servicing
609 animations with back-to-back frame updates.
611 TTimeIntervalMicroSeconds CWindowServer::CDefaultAnimationScheduler::GetDueDelta (TBool aForceRedraw, TSchedule* aScheduledUpdate)
613 WS_ASSERT_DEBUG(aScheduledUpdate, EWsPanicWsGraphic);
614 WS_ASSERT_DEBUG(aScheduledUpdate->iScheduled, EWsPanicWsGraphic);
617 TInt64 grace = I64LIT(0);
618 TTimeIntervalMicroSeconds thisUpdateDueIn = I64LIT(0); //Microseconds from now
620 // Only use grace periods if not forcing due updates.
623 if (aScheduledUpdate->iRedraw)
624 grace = iRedrawGracePeriod;
626 grace = iAnimationGracePeriod;
630 thisUpdateDueIn = aScheduledUpdate->iWhen.MicroSecondsFrom(now);
632 // Add the grace period if the update is due in less time than the grace period.
633 if (thisUpdateDueIn < grace)
634 thisUpdateDueIn = grace;
635 else if (thisUpdateDueIn > KHalfHour)
636 thisUpdateDueIn = KHalfHour;
638 return thisUpdateDueIn;
642 Schedule an actual screen update at the point in time at which it is due. The due time may be modified by
643 this method based on any "grace period" values.
645 @param aScreen Screen number to update.
646 @param aForceRedraw Force redraws that are due. This causes grace periods not to be used.
648 void CWindowServer::CDefaultAnimationScheduler::ScheduleUpdate (TInt aScreenNumber, TBool aForceRedraw)
650 // Schedule updates for any invalidated regions.
651 RedrawAllInvalidatedRegions (aScreenNumber);
653 TSchedule* scheduledUpdate = GetScheduledScreenUpdate(aScreenNumber);
656 WS_ASSERT_DEBUG(scheduledUpdate->iScheduled, EWsPanicWsGraphic);
657 WS_ASSERT_DEBUG(aScreenNumber < iScreenState.Count(), EWsPanicWsGraphic);
659 CScreenState& screenState = *iScreenState[aScreenNumber];
661 // Initiate redraw if scheduled and not currently updating the display.
662 if(!screenState.IsActive())
664 TTimeIntervalMicroSeconds thisUpdateDueIn =
665 GetDueDelta (aForceRedraw, scheduledUpdate);
667 // Reschedule any preexisting update if this one is due earlier.
668 // If this update is not due earlier than a preexisting update then
669 // there is nothing to do - just let the pending update occur.
672 TBool performUpdate = ETrue;
673 if(screenState.iUpdateOn->IsActive())
675 if (thisUpdateDueIn < screenState.iExpectedTickTime.MicroSecondsFrom(now))
676 screenState.iUpdateOn->Cancel();
678 performUpdate = EFalse;
683 if (thisUpdateDueIn.Int64() == 0) // Perform an immediate update if we are due.
685 screenState.iExpectedTickTime = now;
686 InvokeDueAnimation(aScreenNumber);
688 else // Schedule the tick at the appropriate time.
690 WS_ASSERT_DEBUG(thisUpdateDueIn.Int64() > 0, EWsPanicWsGraphic);
691 screenState.iExpectedTickTime = now + thisUpdateDueIn;
692 screenState.iUpdateOn->Start(thisUpdateDueIn.Int64(),0,TCallBack(InvokeDueAnimation, &screenState.iScreenUpdateDetails));
700 @return A pointer to the scheduled update details currently associated with the specified screen.
701 If there is no scheduled update then NULL is returned.
702 @note There is only ever one scheduled update per screen.
704 CWindowServer::CDefaultAnimationScheduler::TSchedule* CWindowServer::CDefaultAnimationScheduler::GetScheduledScreenUpdate(TInt aScreenNumber)
706 TSchedule* result = NULL;
707 const TInt count = iSchedule.Count();
708 for(TInt ii = 0; ii < count; ii++)
710 if (iSchedule[ii].iScreenNumber == aScreenNumber)
712 if (iSchedule[ii].iScheduled)
713 result = &iSchedule[ii];
722 Redraw invalidated graphic IDs. If invalid regions exist, this will cause ScheduleRedraw() to be invoked.
724 void CWindowServer::CDefaultAnimationScheduler::RedrawAllInvalidatedRegions (TInt aScreen)
726 WS_ASSERT_DEBUG(iScreenState.Count() > aScreen, EWsPanicWsGraphic);
728 CScreenState& screenState = *iScreenState[aScreen];
729 if(screenState.iInvalidateAll || screenState.iInvalidated.Count())
731 const TArray<TGraphicDrawerId> invalidArray = screenState.iInvalidated.Array();
732 MWsScreen* screen = iEnv.Screen(aScreen);
733 WS_ASSERT_DEBUG(screen, EWsPanicWsGraphic);
736 if(screenState.iInvalidateAll)
739 RedrawInvalid(*screen, screenState.iInvalidated.Array());
741 screenState.iInvalidateAll = EFalse;
743 screenState.iInvalidated.Reset();
746 TBool CWindowServer::CDefaultAnimationScheduler::InvokeDueAnimation(TAny* aAny)
748 WS_ASSERT_DEBUG(aAny, EWsPanicWsGraphic);
749 TScreenUpdateDetails* args = reinterpret_cast<TScreenUpdateDetails*>(aAny);
751 args->iScheduler->InvokeDueAnimation (args->iScreenNumber);
756 void CWindowServer::CDefaultAnimationScheduler::InvokeDueAnimation(TInt aScreen)
758 WS_ASSERT_DEBUG(aScreen < iScreenState.Count(), EWsPanicWsGraphic);
759 CScreenState& screenState = *iScreenState[aScreen];
760 WS_ASSERT_DEBUG(!screenState.IsActive(), EWsPanicWsGraphic);
762 // All updates are driven through ScheduleRedraw() and ScheduleAnimation().
763 screenState.iUpdateOn->Cancel();
765 TSchedule* scheduledUpdate = GetScheduledScreenUpdate(aScreen);
768 WS_ASSERT_DEBUG(scheduledUpdate->iScheduled, EWsPanicWsGraphic);
770 // Honour any flags that indicate we should not redraw.
771 switch(iInactivityBehaviour)
773 case EStopAnimation :
774 // Stop server side drawing. Only the client may redraw if iInactive is set.
775 if(iInactive && !scheduledUpdate->iRedraw)
777 iInactiveDraws = ETrue;
781 case EStopAllDrawing :
782 // Stop both client and server side drawing.
785 iInactiveDraws = ETrue;
792 WS_ASSERT_DEBUG(EFalse, EWsPanicWsGraphic);
796 scheduledUpdate->iScheduled = EFalse;
797 screenState.SetActive();
798 Animate(*scheduledUpdate->iScreen, &(screenState.iStatus));
802 // CWindowServer::CServer \\\\\\\\\\\\\\\\\\\\\\
804 class CWindowServer::CServer : public CPolicyServer
807 static CServer* NewL()
809 return new(ELeave) CServer;
813 CPolicyServer::StartL(KWSERVServerName);
817 iSessionIter.SetToFirst();
819 while(iSessionIter++)
824 public: //from CPolicyServer
825 /** Creates a new client for this server. */
826 CSession2* NewSessionL(const TVersion& aVersion, const RMessage2& aMessage) const
828 TVersion v(KWservMajorVersionNumber, KWservMinorVersionNumber, KWservBuildVersionNumber);
829 if (User::QueryVersionSupported(v, aVersion)==EFalse)
830 User::Leave(KErrNotSupported);
832 User::LeaveIfError(aMessage.Client(thread));
833 return(new(ELeave) CWsClient(thread));
836 CServer() : CPolicyServer(EMainServerPriority, KWsServPolicy)
841 // CWindowServer \\\\\\\\\\\\\\\\\\\\\\\\\\\
843 CWindowServer *CWindowServer::NewL()
845 // Create a new CWindowServer.
848 CWindowServer* self = new(ELeave) CWindowServer();
849 CleanupStack::PushL(self);
851 CleanupStack::Pop(self);
855 CWindowServer::CWindowServer()
862 CWindowServer::~CWindowServer()
866 iMemoryReleases.Reset();
867 WS_ASSERT_DEBUG(iDrawerMasterIndex.IsEmpty(), EWsPanicWsGraphic);
868 iDrawerMasterIndex.Close();
870 delete iDefaultAnimationScheduler;
871 iDefaultAnimationScheduler = NULL; //might be called from clients during server destruction
874 void CWindowServer::ConstructL()
876 iServer = CServer::NewL();
877 CWsTop::PluginManager()->InitializePluginsL(*this); // plugins are loaded and own by CWsTop
878 iDefaultAnimationScheduler = new(ELeave) CDefaultAnimationScheduler(*this);
879 iDefaultAnimationScheduler->ConstructL();
880 RegisterMemoryRelease(this);
883 void CWindowServer::StartL()
888 void CWindowServer::SetPinClientDescriptors(TBool aPin)
890 iServer->SetPinClientDescriptors(aPin);
893 TInt CWindowServer::SessionCount()
895 return iServer->SessionCount();
898 const CWsGraphicDrawer* CWindowServer::ResolveGraphic(const TGraphicDrawerId& aId) const
900 return iDrawerMasterIndex.ResolveGraphic(aId);
903 void CWindowServer::Invalidate(const TGraphicDrawerId& aId)
905 AnimationScheduler()->Invalidate(aId);
908 TInt CWindowServer::ScreenCount() const
910 return CWsTop::NumberOfScreens();
913 MWsScreen* CWindowServer::Screen(TInt aIndex)
915 if((aIndex >= 0) && (aIndex < ScreenCount()))
917 return CWsTop::Screen(aIndex);
922 const MWsScreen* CWindowServer::Screen(TInt aIndex) const
924 if((aIndex >= 0) && (aIndex < ScreenCount()))
926 return CWsTop::Screen(aIndex);
932 Custom Animation Scheduler
934 TBool CWindowServer::SetCustomAnimationScheduler(MWsAnimationScheduler* /*aScheduler*/)
939 TBool CWindowServer::HasCustomAnimationScheduler() const
944 TBool CWindowServer::ClearCustomAnimationScheduler(MWsAnimationScheduler* /*aCurrentScheduler*/)
949 MWsAnimationScheduler* CWindowServer::AnimationScheduler()
951 return iDefaultAnimationScheduler;
954 void CWindowServer::PrepareShutdown()
956 //Stop the renderloop, i.e. prevent any further calls to MWsAnimationScheduler::Animate()
957 delete iDefaultAnimationScheduler;
958 iDefaultAnimationScheduler = NULL;
961 TInt CWindowServer::RegisterEventHandler(CWsGraphicDrawer* aDrawer, MWsEventHandler* aHandler, TUint32 aEventMask)
963 if (!aDrawer || !aHandler || aEventMask==0)
965 TInt err = TWindowServerEvent::RegisterDrawerHandler(aDrawer, aEventMask);
968 aDrawer->SetEventHandler(aHandler);
972 TInt CWindowServer::UnregisterEventHandler(CWsGraphicDrawer* aDrawer)
974 if (!aDrawer || (aDrawer && !aDrawer->HasEventHandler()))
976 TInt err = TWindowServerEvent::UnregisterDrawerHandler(aDrawer);
979 aDrawer->SetEventHandler(NULL);
983 TInt CWindowServer::RegisterWsEventHandler(MWsEventHandler* aHandler, TUint32 aEventMask)
985 if (!aHandler || aEventMask==0)
987 return TWindowServerEvent::RegisterWsEventHandler(aHandler, aEventMask);
990 TInt CWindowServer::UnregisterWsEventHandler(MWsEventHandler* aHandler)
992 return TWindowServerEvent::UnregisterWsEventHandler(aHandler);
995 TAny* CWindowServer::ResolveObjectInterface(TUint aTypeId)
999 case MWsActiveSchedulerDebug::EWsObjectInterfaceId:
1000 return static_cast<MWsActiveSchedulerDebug*>(CWsActiveScheduler::Static());
1001 case MWsIniFile::EWsObjectInterfaceId:
1002 return static_cast<MWsIniFile*>(WsIniFile);
1005 if (CWsPluginManager *plugMgr=CWsTop::PluginManager())
1006 return plugMgr->ResolveObjectInterface(aTypeId);
1011 void CWindowServer::Log(TInt aPriority,const TDesC &aFmt,TInt aParam)
1015 wsDebugLog->MiscMessage(aPriority, aFmt, aParam);
1019 // CWsGraphicDrawer master index
1021 TInt CWindowServer::AddGraphicDrawer(CWsGraphicDrawer* aDrawer)
1023 return iDrawerMasterIndex.Add(aDrawer);
1026 TInt CWindowServer::SwapGraphicDrawer(CWsGraphicDrawer* aDrawer)
1028 return iDrawerMasterIndex.Swap(aDrawer);
1031 TInt CWindowServer::RemoveGraphicDrawer(const TGraphicDrawerId& aId)
1033 return iDrawerMasterIndex.Remove(aId);
1036 TInt CWindowServer::RemoveAllGraphicDrawers(const MWsClient& aOwner)
1038 return iDrawerMasterIndex.RemoveAll(aOwner);
1041 TInt CWindowServer::RegisterMemoryRelease(MWsMemoryRelease * aMemoryRelease)
1043 return iMemoryReleases.Append(aMemoryRelease);
1046 void CWindowServer::UnregisterMemoryRelease(MWsMemoryRelease * aMemoryRelease)
1048 for (TInt ii = iMemoryReleases.Count() - 1; ii >= 0; --ii)
1050 if (iMemoryReleases[ii] == aMemoryRelease)
1052 iMemoryReleases.Remove(ii);
1058 TBool CWindowServer::ReleaseMemory(TMemoryReleaseLevel aLevel)
1060 return CWsWindow::ReleaseMemory(aLevel);
1063 TBool CWindowServer::ReleaseMemory()
1065 TBool released = EFalse;
1066 for (TInt level = MWsMemoryRelease::ELow; !released && level <= MWsMemoryRelease::EHigh; ++level)
1068 for (TInt ii = iMemoryReleases.Count() - 1; !released && ii >= 0; --ii)
1070 released = iMemoryReleases[ii]->ReleaseMemory(static_cast<TMemoryReleaseLevel>(level));
1076 void CWindowServer::DestroySessionsForShutdown()