First public contribution.
1 // Copyright (c) 2007-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.
21 static const TTimeIntervalMicroSeconds KBurstDuration = 1000000; //one second
22 static const TInt KMaxMemoryReleasesPerBurst = 5;
23 CWsMemoryManager * CWsMemoryManager::iStatic = NULL;
25 CWsMemoryManager * CWsMemoryManager::Static()
30 CWsMemoryManager * CWsMemoryManager::NewLC()
32 CWsMemoryManager * self = new (ELeave) CWsMemoryManager;
33 CleanupStack::PushL(self);
39 CWsMemoryManager::CWsMemoryManager()
41 iImpl = User::SwitchAllocator(this);
44 CWsMemoryManager::~CWsMemoryManager()
46 WS_ASSERT_ALWAYS(this == User::SwitchAllocator(iImpl),EWsPanicMemoryManager);
54 void CWsMemoryManager::ConstructL()
56 _LIT(KMemMgrReserve, "MEMORYRESERVE");
57 const TInt KDefaultMemMgrReserve = 1024;
59 if (!WsIniFile->FindVar(KMemMgrReserve, iReserveSize))
60 iReserveSize = KDefaultMemMgrReserve;
63 iReserve = Alloc(iReserveSize);
65 iCurrentBurstStart.UniversalTime();
69 Tells the memory manager to fail on next retry.
70 I.e. whenever the next allocation failure occurs, the memory manager won't try
71 to free up memory to have another go at the allocation.
73 N.B. this only applies for the next failure, the state is reset when there is an
76 This method is only to be used for OOM testing.
78 void CWsMemoryManager::SetFailNextRetry()
80 iFailNextRetry = ETrue;
84 Implementing RAllocator
88 Alloc and ReAlloc attempt to obtain memory through CWsTop::ReleaseMemory when they run low.
89 ReleaseMemory looks for blocks of memory that the window server doesn't need urgently and frees
92 TAny* CWsMemoryManager::Alloc(TInt aSize)
94 TBool keepTrying = ETrue;
98 return iImpl->Alloc(aSize); //fallback on RAllocator
100 if(TAny* ret = iImpl->Alloc(aSize)) //normal case
105 if(now.MicroSecondsFrom(iCurrentBurstStart) < KBurstDuration)
107 iCurrentBurstReleaseCount++;
108 if(iCurrentBurstReleaseCount > KMaxMemoryReleasesPerBurst)
113 iCurrentBurstStart = now;
114 iCurrentBurstReleaseCount = 1;
117 if(iReserveEnabled && iReserve && (aSize < iReserveSize))
125 keepTrying = CWsTop::ReleaseMemory();
128 const TInt reclaimed = Compress(); //Try to give back to the OS
133 //used for OOM testing only
136 iFailNextRetry = EFalse;
145 TAny* CWsMemoryManager::ReAlloc(TAny* aPtr, TInt aSize, TInt aMode)
147 TBool keepTrying = ETrue;
151 return iImpl->ReAlloc(aPtr, aSize, aMode); //fallback on RAllocator
153 if(TAny* ret = iImpl->ReAlloc(aPtr, aSize, aMode)) //normal case
158 if(now.MicroSecondsFrom(iCurrentBurstStart) < KBurstDuration)
160 iCurrentBurstReleaseCount++;
161 if(iCurrentBurstReleaseCount > KMaxMemoryReleasesPerBurst)
166 iCurrentBurstStart = now;
167 iCurrentBurstReleaseCount = 1;
170 if(iReserveEnabled && iReserve && (aSize < iReserveSize))
178 keepTrying = CWsTop::ReleaseMemory();
181 const TInt reclaimed = Compress(); //Try to give back to the OS
186 //used for OOM testing only
189 iFailNextRetry = EFalse;
199 The rest of these functions just call the default implementation
201 void CWsMemoryManager::Free(TAny* aPtr)
203 return iImpl->Free(aPtr);
206 TInt CWsMemoryManager::AllocLen(const TAny* aCell) const
208 return iImpl->AllocLen(aCell);
211 TInt CWsMemoryManager::Compress()
213 return iImpl->Compress();
216 void CWsMemoryManager::Reset()
221 TInt CWsMemoryManager::AllocSize(TInt& aTotalAllocSize) const
223 return iImpl->AllocSize(aTotalAllocSize);
226 TInt CWsMemoryManager::Available(TInt& aBiggestBlock) const
228 return iImpl->Available(aBiggestBlock);
231 TInt CWsMemoryManager::DebugFunction(TInt aFunc, TAny* a1, TAny* a2)
233 return iImpl->DebugFunction(aFunc,a1,a2);
236 TInt CWsMemoryManager::Count() const
238 return iImpl->Count();
240 /** This is a fairly dumb way to enable and disable the reserve, but we normally
241 get away with it because wserv is high priority. A better approach would be to
242 use placement new into the reserve memory and manage it directly. This would also
243 allow us to track misbehaving code which allocated during OOM drawing and didn't
246 void CWsMemoryManager::EnableReserve()
248 WS_ASSERT_DEBUG(!iReserveEnabled, EWsPanicMemoryManager);
249 iReserveEnabled = ETrue;
252 void CWsMemoryManager::DisableReserve()
254 WS_ASSERT_DEBUG(iReserveEnabled, EWsPanicMemoryManager);
255 iReserveEnabled = EFalse;
256 if((!iReserve) && (iReserveSize > 0))
257 iReserve = Alloc(iReserveSize);