Update contrib.
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"
27 GLREF_D CDebugLogBase *wsDebugLog;
29 const TUint KRangeCount = 1;
30 // We use a lot of 64 bit time calculations, but a periodic can only cope with signed 32 bit times
31 // which gives them a limit of about 35 minutes.
32 // Fortunately, our animtions are safe if redrawn early. Every half an hour isn't going to hurt.
33 const TTimeIntervalMicroSeconds KHalfHour = 30 * 60 * 1000 * 1000;
35 const TInt KWsServRanges[KRangeCount] =
40 const TUint8 KElementsIndex[KRangeCount] =
42 CPolicyServer::EAlwaysPass,
45 const CPolicyServer::TPolicyElement KPolicyElements[] =
47 {_INIT_SECURITY_POLICY_C1(ECapabilityPowerMgmt), CPolicyServer::EFailClient},
48 {_INIT_SECURITY_POLICY_C1(ECapabilitySwEvent), CPolicyServer::EFailClient},
49 {_INIT_SECURITY_POLICY_C1(ECapabilityWriteDeviceData), CPolicyServer::EFailClient}
52 const CPolicyServer::TPolicy KWsServPolicy =
54 CPolicyServer::EAlwaysPass,
61 // CWindowServer::CDefaultAnimationScheduler \\\\\\\\\\\\\\\\\\\\\\\\\\\
63 class CWindowServer::CDefaultAnimationScheduler: public CBase, public MWsAnimationScheduler
66 enum TInactivityBehaviour
73 CDefaultAnimationScheduler(MWsGraphicDrawerEnvironment& aEnv);
74 ~CDefaultAnimationScheduler();
75 void ConstructL(); //LeaveScan: member of nested class declaration
76 // implementing MWsAnimationScheduler
77 void ScheduleAnimation(MWsScreen& aScreen,const TTime& aWhen);
78 void UnscheduleAnimation(MWsScreen& aScreen);
79 void Invalidate(const TGraphicDrawerId& aId);
82 void ScheduleRedraw(MWsScreen& aScreen,const TTime& aWhen);
83 void DoRedrawNow(MWsScreen& aScreen);
84 void DoRedrawNow(MWsScreen& aScreen, MWsAnimationScheduler::MScreenUpdateObserver& aObserver);
85 void ClearScreenUpdateObserver(const MWsAnimationScheduler::MScreenUpdateObserver& /*aObserver*/) {}
86 TInt RemoveGraphicDrawer(const TGraphicDrawerId &aId);
89 static TBool OnIdleCallBack(TAny* aAny);
90 void OnIdleCallBack(TBool aForceRedraw);
91 static TBool OnTickCallBack(TAny* aAny);
92 void OnTickCallBack();
93 static TBool OnKickBack(TAny* aAny);
96 MWsGraphicDrawerEnvironment& iEnv;
97 static const TInt64 KRedrawGrace;
98 static const TInt64 KAnimationGrace;
99 static const TInt64 KDsaAnimationGrace;
100 CAsyncCallBack* iIdleInitiator;
102 CKickBack* iKickBack;
105 MWsScreen* iScreen; // used as a unique index, searching with FindInUnsignedKeyOrder
110 RArray<TSchedule> iSchedule;
112 RArray<TGraphicDrawerId> iInvalidated;
113 TBool iInvalidateAll; // if we could not add to iInvalidated, we have to instead redraw everything
116 TBool iInactiveDraws;
118 TBool iRedrawScheduled;
119 TInt64 iRedrawGracePeriod;
120 TInt64 iAnimationGracePeriod;
121 TInt64 iDsaAnimationGracePeriod;
122 TInactivityBehaviour iInactivityBehaviour;
125 TInt CWindowServer::CDefaultAnimationScheduler::RemoveGraphicDrawer(const TGraphicDrawerId &aId)
127 TInt index=iInvalidated.Find(aId);
128 if (index!=KErrNotFound)
129 iInvalidated.Remove(index);
133 class CWindowServer::CDefaultAnimationScheduler::CKickBack: public CActive
136 CKickBack(const TCallBack& aCallBack);
138 void RequestKickBack();
141 static TInt IdleThreadFunc(TAny* aAny);
144 void RunL(); // fires when kicked back by the idle thread
147 RThread iWservThread;
149 TRequestStatus iIdleStatus;
153 CWindowServer::CDefaultAnimationScheduler::CKickBack::CKickBack(const TCallBack& aCallBack) :
154 CActive(EPriorityNormal),
157 CActiveScheduler::Add(this);
160 void CWindowServer::CDefaultAnimationScheduler::CKickBack::ConstructL()
162 _LIT(KIdleThreadName,"NearlyIdleKickBack");
163 const TInt KStackSize = 1024;
164 User::LeaveIfError(iWservThread.Open(iWservThread.Id()));
165 User::LeaveIfError(iIdleThread.Create(KIdleThreadName(),IdleThreadFunc,KStackSize,NULL,this));
166 iIdleThread.SetPriority(EPriorityAbsoluteVeryLow);
167 iIdleThread.Resume();
170 void CWindowServer::CDefaultAnimationScheduler::CKickBack::RequestKickBack()
174 iStatus = KRequestPending;
176 TRequestStatus * status = &iIdleStatus;
177 iIdleThread.RequestComplete(status, KErrNone);
181 void CWindowServer::CDefaultAnimationScheduler::CKickBack::Loop()
185 // This is used here for performance reasons.
186 User::WaitForRequest(iIdleStatus);
187 iIdleStatus = KRequestPending;
188 if (IsActive()&& (iStatus == KRequestPending))
190 TRequestStatus * status = &iStatus;
191 iWservThread.RequestComplete(status,KErrNone);
196 void CWindowServer::CDefaultAnimationScheduler::CKickBack::RunL()
198 iCallBack.CallBack();
201 void CWindowServer::CDefaultAnimationScheduler::CKickBack::DoCancel()
205 CWindowServer::CDefaultAnimationScheduler::CKickBack::~CKickBack()
208 iWservThread.Close();
213 TInt CWindowServer::CDefaultAnimationScheduler::CKickBack::IdleThreadFunc(TAny* aAny)
215 CKickBack* owner = reinterpret_cast<CKickBack*>(aAny);
224 // If using the default animation scheduler on a device, these two numbers may be worth tweaking in the inifile
225 // However, both are maximum periods - wserv will go faster than either if nothing else is using the system.
226 const TInt64 CWindowServer::CDefaultAnimationScheduler::KRedrawGrace = 0; // do redraws immediately
227 const TInt64 CWindowServer::CDefaultAnimationScheduler::KAnimationGrace = 35000; // insist upon 35ms grace for other threads to run when animating
228 const TInt64 CWindowServer::CDefaultAnimationScheduler::KDsaAnimationGrace = 35000; // insist upon 35ms grace for other threads to run when animating during DSA
230 CWindowServer::CDefaultAnimationScheduler::CDefaultAnimationScheduler(MWsGraphicDrawerEnvironment& aEnv):
231 iEnv(aEnv), iSchedule(1,_FOFF(TSchedule,iScreen))
235 CWindowServer::CDefaultAnimationScheduler::~CDefaultAnimationScheduler()
238 iInvalidated.Close();
240 delete iIdleInitiator;
244 void CWindowServer::CDefaultAnimationScheduler::ConstructL()
246 _LIT(KOnInactive,"ONINACTIVE");
247 _LIT(KStopAnimation,"STOPANIMATION");
248 _LIT(KStopAllDrawing,"STOPALLDRAWING");
249 _LIT(KIgnore,"IGNORE");
251 TPtrC inactivityBehaviourString;
252 WsIniFile->FindVar(KOnInactive,inactivityBehaviourString);
253 if(inactivityBehaviourString.CompareF(KStopAnimation)==0)
254 iInactivityBehaviour = EStopAnimation;
255 else if(inactivityBehaviourString.CompareF(KStopAllDrawing)==0)
256 iInactivityBehaviour = EStopAllDrawing;
257 else if(inactivityBehaviourString.CompareF(KIgnore)==0)
258 iInactivityBehaviour = EIgnore;
260 _LIT(KRedrawGracePeriod, "REDRAWGRACEPERIOD");
261 TInt tmp = KRedrawGrace;
262 WsIniFile->FindVar(KRedrawGracePeriod, tmp);
263 iRedrawGracePeriod = tmp;
265 _LIT(KAnimationGracePeriod, "ANIMATIONGRACEPERIOD");
266 tmp = KAnimationGrace;
267 WsIniFile->FindVar(KAnimationGracePeriod, tmp);
268 iAnimationGracePeriod = tmp;
270 _LIT(KDsaAnimationGracePeriod, "DSAANIMATIONGRACEPERIOD");
271 tmp = KDsaAnimationGrace;
272 WsIniFile->FindVar(KDsaAnimationGracePeriod, tmp);
273 iDsaAnimationGracePeriod = tmp;
275 iIdleInitiator = new(ELeave) CAsyncCallBack(TCallBack(OnIdleCallBack,this),EWsGraphicAnimateAwaitIdlePriority);
276 iTick = CPeriodic::NewL(EWsGraphicAnimatePriority);
278 _LIT(KDisableIdleAnimation, "DISABLEIDLEANIMATION");
279 if (!WsIniFile->FindVar(KDisableIdleAnimation))
281 iKickBack = new CKickBack(TCallBack(OnKickBack,this));
282 iKickBack->ConstructL();
286 void CWindowServer::CDefaultAnimationScheduler::Invalidate(const TGraphicDrawerId& aId)
290 switch(iInvalidated.InsertInOrder(aId,TLinearOrder<TGraphicDrawerId>(TGraphicDrawerId::Compare)))
294 case KErrAlreadyExists:
297 iInvalidateAll = ETrue;
298 iInvalidated.Reset();
301 iIdleInitiator->CallBack();
304 void CWindowServer::CDefaultAnimationScheduler::OnInactive()
309 void CWindowServer::CDefaultAnimationScheduler::OnActive()
314 iInactiveDraws = EFalse;
315 iIdleInitiator->CallBack();
319 void CWindowServer::CDefaultAnimationScheduler::ScheduleRedraw(MWsScreen& aScreen,const TTime& aWhen)
321 iRedrawScheduled = ETrue;
322 ScheduleAnimation(aScreen, aWhen);
325 void CWindowServer::CDefaultAnimationScheduler::DoRedrawNow(MWsScreen& /*aScreen*/)
327 OnIdleCallBack(ETrue);
330 void CWindowServer::CDefaultAnimationScheduler::DoRedrawNow(MWsScreen& /*aScreen*/, MWsAnimationScheduler::MScreenUpdateObserver& aObserver)
332 OnIdleCallBack(ETrue);
333 aObserver.ScreenUpdateComplete(KErrNone);
336 void CWindowServer::CDefaultAnimationScheduler::ScheduleAnimation(MWsScreen& aScreen,const TTime& aWhen)
339 schedule.iScreen = &aScreen;
340 schedule.iSchedule = ETrue;
341 schedule.iWhen = aWhen;
342 schedule.iGeneration = iGeneration;
344 const TInt idx = iSchedule.FindInUnsignedKeyOrder(schedule);
347 if(iSchedule[idx].iSchedule)
349 if(iSchedule[idx].iWhen > aWhen)
351 iSchedule[idx].iWhen = aWhen;
356 iSchedule[idx] = schedule;
358 iSchedule[idx].iGeneration = iGeneration;
363 ok = (KErrNone == iSchedule.InsertInUnsignedKeyOrder(schedule));
367 //If the animation runs at very high rate which exceeds the rate WSERV can
368 //perform the rendering, it is possible that WSERV animation loop will monopolize
370 User::After(0); // to yeild from the animation loop
372 iIdleInitiator->CallBack();
376 void CWindowServer::CDefaultAnimationScheduler::UnscheduleAnimation(MWsScreen& aScreen)
379 schedule.iScreen = &aScreen;
380 const TInt idx = iSchedule.FindInUnsignedKeyOrder(schedule);
383 iSchedule[idx].iSchedule = EFalse;
387 TBool CWindowServer::CDefaultAnimationScheduler::OnIdleCallBack(TAny* aAny)
389 CDefaultAnimationScheduler* self = reinterpret_cast<CDefaultAnimationScheduler*>(aAny);
393 self->OnIdleCallBack(EFalse);
395 return EFalse; //ignored by caller
398 void CWindowServer::CDefaultAnimationScheduler::OnIdleCallBack(TBool aForceRedraw)
400 TBool wasActive = EFalse;
401 if (iTick->IsActive())
404 iTick->Cancel(); // stop ticker, as we'll reschedule if necessary
408 // Don't need to update iExpectedTime as we will not schedule a tick.
409 // Don't need to update iWhenDesired as we will not schedule a kick-back.
413 TBool tick = (iInvalidateAll || iInvalidated.Count());
414 TTimeIntervalMicroSeconds next = 0LL;
415 const TInt count = iSchedule.Count();
420 // work out the next wanted tick
421 TBool animTick = EFalse;
422 for(TInt i=0; i<count; i++)
424 if(iSchedule[i].iSchedule)
426 const TTime when = iSchedule[i].iWhen;
427 const TTimeIntervalMicroSeconds fromNow = when.MicroSecondsFrom(now);
428 if(!animTick || (fromNow < next))
441 if (iRedrawScheduled)
442 grace = iRedrawGracePeriod;
443 else if (EFalse) // DSA active
444 grace = iDsaAnimationGracePeriod;
446 grace = iAnimationGracePeriod;
449 TInt64 minimum = iExpectedTime.MicroSecondsFrom(now).Int64();
450 if (minimum >= 0 && grace > minimum)
454 if (next.Int64() <= 0)
455 next = 0LL ; // No kickback/tick is needed. Make sure next == default grace period.
461 iKickBack->RequestKickBack();
463 else if (next > KHalfHour)
466 iExpectedTime=now + next;
467 if (next.Int64() > 0)
469 iTick->Start(next.Int64(),0,TCallBack(OnTickCallBack,this));
473 OnTickCallBack(); // scheduling for 0 doesn't actually execute immediately
478 TBool CWindowServer::CDefaultAnimationScheduler::OnKickBack(TAny* aAny)
480 CDefaultAnimationScheduler* self = reinterpret_cast<CDefaultAnimationScheduler*>(aAny);
486 return EFalse; //ignored by caller
489 void CWindowServer::CDefaultAnimationScheduler::OnKickBack()
491 if (iTick->IsActive())
497 TTimeIntervalMicroSeconds fromNow = iWhenDesired.MicroSecondsFrom(now);
505 if (fromNow > KHalfHour)
507 iTick->Start(fromNow.Int64(),0,TCallBack(OnTickCallBack, this));
512 TBool CWindowServer::CDefaultAnimationScheduler::OnTickCallBack(TAny* aAny)
514 CDefaultAnimationScheduler* self = reinterpret_cast<CDefaultAnimationScheduler*>(aAny);
518 self->OnTickCallBack();
523 void CWindowServer::CDefaultAnimationScheduler::OnTickCallBack()
527 switch(iInactivityBehaviour)
529 case EStopAnimation :
530 // only client redraws if inactive. server side drawing stopped.
531 if(iInactive && !iRedrawScheduled)
533 iInactiveDraws = ETrue;
537 case EStopAllDrawing :
538 // if screen off, stop both client and server side drawing.
541 iInactiveDraws = ETrue;
547 // ignore inactivity and draw as normal
551 iRedrawScheduled = EFalse;
555 CWsActiveScheduler::Static()->AccumReclaimedIdleTime(Max(0LL,now.MicroSecondsFrom(iExpectedTime).Int64()));
558 TInt scheduledCount = 0;
559 /* first redraw any screens that are affected by invalidated graphic IDs */
560 if(iInvalidateAll || iInvalidated.Count())
562 // cancel idle callback if it's already scheduled as we're going to redraw all invalidated
563 // request at this point and clear invalidated array
564 if (iIdleInitiator->IsActive())
565 iIdleInitiator->Cancel();
566 const TArray<TGraphicDrawerId> invalidArray = iInvalidated.Array();
567 const TInt screenCount = iEnv.ScreenCount();
568 for(TInt i=0; i<screenCount; i++)
570 MWsScreen* screen = iEnv.Screen(i);
580 RedrawInvalid(*screen,invalidArray);
584 iInvalidateAll = EFalse;
585 iInvalidated.Reset();
587 /* iSchedule might resize during this call because an Animate() on a screen might reschedule another animation etc */
588 TInt generation = iGeneration++;
593 const TInt count = iSchedule.Count();
594 for(TInt i=0; i<count; i++)
596 if(iSchedule[i].iSchedule)
599 if(iSchedule[i].iGeneration == generation)
601 iSchedule[i].iSchedule = EFalse; // denote not to tick; it can be rescheduled in the subsequent call to Animate()
602 Animate(*(iSchedule[i].iScreen));
605 break; // go back to outer do while(drew)
611 TBool idleIsActive = iIdleInitiator->IsActive();
612 if(scheduledCount || !drewCount)
614 TBool rescheduled = EFalse;
615 for(TInt i=0; i<iSchedule.Count(); i++)
617 rescheduled |= iSchedule[i].iSchedule;
626 // CWindowServer \\\\\\\\\\\\\\\\\\\\\\\\\\\
628 CWindowServer *CWindowServer::NewL()
630 // Create a new CWindowServer.
633 CWindowServer* self = new(ELeave) CWindowServer();
634 CleanupStack::PushL(self);
636 CleanupStack::Pop(self);
640 CWindowServer::CWindowServer()
644 : CPolicyServer(EMainServerPriority,KWsServPolicy)
648 CWindowServer::~CWindowServer()
650 iMemoryReleases.Reset();
651 WS_ASSERT_DEBUG(iDrawerMasterIndex.IsEmpty(), EWsPanicWsGraphic);
652 iDrawerMasterIndex.Close();
654 delete iDefaultAnimationScheduler;
655 iDefaultAnimationScheduler = NULL; // might be called from clients during server destruction
656 delete iPluginManager;
659 void CWindowServer::ConstructL()
661 iPluginManager = CWsPluginManager::NewL(*this);
662 iDefaultAnimationScheduler = new(ELeave) CDefaultAnimationScheduler(*this);
663 iDefaultAnimationScheduler->ConstructL();
664 RegisterMemoryRelease(this);
667 /** Creates a new client for this server. */
668 CSession2* CWindowServer::NewSessionL(const TVersion& aVersion,const RMessage2& aMessage) const
670 TVersion v(KWservMajorVersionNumber,KWservMinorVersionNumber,KWservBuildVersionNumber);
671 if (User::QueryVersionSupported(v,aVersion)==EFalse)
672 User::Leave(KErrNotSupported);
674 User::LeaveIfError(aMessage.Client(thread));
675 return(new(ELeave) CWsClient(thread));
678 TInt CWindowServer::SessionCount()
680 iSessionIter.SetToFirst();
682 while(iSessionIter++)
687 const CWsGraphicDrawer* CWindowServer::ResolveGraphic(const TGraphicDrawerId& aId) const
689 return iDrawerMasterIndex.ResolveGraphic(aId);
692 void CWindowServer::Invalidate(const TGraphicDrawerId& aId)
694 AnimationScheduler()->Invalidate(aId);
697 TInt CWindowServer::ScreenCount() const
699 return CWsTop::NumberOfScreens();
702 MWsScreen* CWindowServer::Screen(TInt aIndex)
704 if((aIndex >= 0) && (aIndex < ScreenCount()))
706 return CWsTop::Screen(aIndex);
711 const MWsScreen* CWindowServer::Screen(TInt aIndex) const
713 if((aIndex >= 0) && (aIndex < ScreenCount()))
715 return CWsTop::Screen(aIndex);
721 Custom Animation Scheduler
723 TBool CWindowServer::SetCustomAnimationScheduler(MWsAnimationScheduler* /*aScheduler*/)
728 TBool CWindowServer::HasCustomAnimationScheduler() const
733 TBool CWindowServer::ClearCustomAnimationScheduler(MWsAnimationScheduler* /*aCurrentScheduler*/)
738 MWsAnimationScheduler* CWindowServer::AnimationScheduler()
740 return iDefaultAnimationScheduler;
743 TInt CWindowServer::RegisterEventHandler(CWsGraphicDrawer* aDrawer, MWsEventHandler* aHandler, TUint32 aEventMask)
745 if (!aDrawer || !aHandler || aEventMask==0)
747 TInt err = TWindowServerEvent::RegisterDrawerHandler(aDrawer, aEventMask);
750 aDrawer->SetEventHandler(aHandler);
754 TInt CWindowServer::UnregisterEventHandler(CWsGraphicDrawer* aDrawer)
756 if (!aDrawer || (aDrawer && !aDrawer->HasEventHandler()))
758 TInt err = TWindowServerEvent::UnregisterDrawerHandler(aDrawer);
761 aDrawer->SetEventHandler(NULL);
765 TInt CWindowServer::RegisterWsEventHandler(MWsEventHandler* aHandler, TUint32 aEventMask)
767 if (!aHandler || aEventMask==0)
769 return TWindowServerEvent::RegisterWsEventHandler(aHandler, aEventMask);
772 TInt CWindowServer::UnregisterWsEventHandler(MWsEventHandler* aHandler)
774 return TWindowServerEvent::UnregisterWsEventHandler(aHandler);
777 TInt CWindowServer::RegisterRawEventHandler(MEventHandler* aHandler)
781 TRAPD(err, TWindowServerEvent::PotentialEventHandlerL(1));
784 TWindowServerEvent::AddEventHandler(aHandler);
788 void CWindowServer::UnregisterRawEventHandler(MEventHandler* aHandler)
790 TWindowServerEvent::RemoveEventHandler(aHandler);
791 TWindowServerEvent::PotentialEventHandlerL(-1); // can't leave for -1
794 void CWindowServer::PostRawEvent(const TRawEvent & aEvent)
796 TWindowServerEvent::ProcessRawEvent(aEvent);
799 void CWindowServer::PostKeyEvent(const TKeyEvent & aEvent)
801 TWindowServerEvent::ProcessKeyEvent(aEvent, 0);
804 TAny* CWindowServer::ResolveObjectInterface(TUint aTypeId)
808 case MWsActiveSchedulerDebug::EWsObjectInterfaceId:
809 return static_cast<MWsActiveSchedulerDebug*>(CWsActiveScheduler::Static());
810 case MWsIniFile::EWsObjectInterfaceId:
811 return static_cast<MWsIniFile*>(WsIniFile);
812 case MWsRawEventServer::EWsObjectInterfaceId:
813 return static_cast<MWsRawEventServer*>(this);
817 return iPluginManager->ResolveObjectInterface(aTypeId);
822 void CWindowServer::Log(TInt aPriority,const TDesC &aFmt,TInt aParam)
826 wsDebugLog->MiscMessage(aPriority, aFmt, aParam);
830 // CWsGraphicDrawer master index
832 TInt CWindowServer::AddGraphicDrawer(CWsGraphicDrawer* aDrawer)
834 return iDrawerMasterIndex.Add(aDrawer);
837 TInt CWindowServer::SwapGraphicDrawer(CWsGraphicDrawer* aDrawer)
839 return iDrawerMasterIndex.Swap(aDrawer);
842 TInt CWindowServer::RemoveGraphicDrawer(const TGraphicDrawerId& aId)
844 iDefaultAnimationScheduler->RemoveGraphicDrawer(aId);
845 return iDrawerMasterIndex.Remove(aId);
848 TInt CWindowServer::RemoveAllGraphicDrawers(const MWsClient& aOwner)
850 return iDrawerMasterIndex.RemoveAll(aOwner);
853 CWsPluginManager* CWindowServer::PluginManager()
855 return iPluginManager;
858 TInt CWindowServer::RegisterMemoryRelease(MWsMemoryRelease * aMemoryRelease)
860 return iMemoryReleases.Append(aMemoryRelease);
863 void CWindowServer::UnregisterMemoryRelease(MWsMemoryRelease * aMemoryRelease)
865 for (TInt i = iMemoryReleases.Count() - 1; i >= 0; --i)
867 if (iMemoryReleases[i] == aMemoryRelease)
869 iMemoryReleases.Remove(i);
875 TBool CWindowServer::ReleaseMemory(TMemoryReleaseLevel aLevel)
877 return CWsWindow::ReleaseMemory(aLevel);
880 TBool CWindowServer::ReleaseMemory()
882 TBool released = EFalse;
883 for (TInt level = MWsMemoryRelease::ELow; !released && level <= MWsMemoryRelease::EHigh; ++level)
885 for (TInt i = iMemoryReleases.Count() - 1; !released && i >= 0; --i)
887 released = iMemoryReleases[i]->ReleaseMemory(static_cast<TMemoryReleaseLevel>(level));