1.1 --- /dev/null Thu Jan 01 00:00:00 1970 +0000
1.2 +++ b/os/graphics/windowing/windowserver/nga/SERVER/WsMemMgr.cpp Fri Jun 15 03:10:57 2012 +0200
1.3 @@ -0,0 +1,258 @@
1.4 +// Copyright (c) 2007-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 +//
1.18 +
1.19 +#include "WsMemMgr.h"
1.20 +#include "inifile.h"
1.21 +#include "panics.h"
1.22 +#include "wstop.h"
1.23 +
1.24 +static const TTimeIntervalMicroSeconds KBurstDuration = 1000000; //one second
1.25 +static const TInt KMaxMemoryReleasesPerBurst = 5;
1.26 +CWsMemoryManager * CWsMemoryManager::iStatic = NULL;
1.27 +
1.28 +CWsMemoryManager * CWsMemoryManager::Static()
1.29 + {
1.30 + return iStatic;
1.31 + }
1.32 +
1.33 +CWsMemoryManager * CWsMemoryManager::NewLC()
1.34 + {
1.35 + CWsMemoryManager * self = new (ELeave) CWsMemoryManager;
1.36 + CleanupStack::PushL(self);
1.37 + self->ConstructL();
1.38 + iStatic = self;
1.39 + return iStatic;
1.40 + }
1.41 +
1.42 +CWsMemoryManager::CWsMemoryManager()
1.43 + {
1.44 + iImpl = User::SwitchAllocator(this);
1.45 + }
1.46 +
1.47 +CWsMemoryManager::~CWsMemoryManager()
1.48 + {
1.49 + WS_ASSERT_ALWAYS(this == User::SwitchAllocator(iImpl),EWsPanicMemoryManager);
1.50 + iStatic = 0;
1.51 + if (iReserve!=NULL)
1.52 + {
1.53 + Free(iReserve);
1.54 + }
1.55 + }
1.56 +
1.57 +void CWsMemoryManager::ConstructL()
1.58 + {
1.59 + _LIT(KMemMgrReserve, "MEMORYRESERVE");
1.60 + const TInt KDefaultMemMgrReserve = 1024;
1.61 +
1.62 + if (!WsIniFile->FindVar(KMemMgrReserve, iReserveSize))
1.63 + iReserveSize = KDefaultMemMgrReserve;
1.64 +
1.65 + if (iReserveSize > 0)
1.66 + iReserve = Alloc(iReserveSize);
1.67 +
1.68 + iCurrentBurstStart.UniversalTime();
1.69 + }
1.70 +
1.71 +/**
1.72 +Tells the memory manager to fail on next retry.
1.73 +I.e. whenever the next allocation failure occurs, the memory manager won't try
1.74 +to free up memory to have another go at the allocation.
1.75 +
1.76 +N.B. this only applies for the next failure, the state is reset when there is an
1.77 +allocation failure.
1.78 +
1.79 +This method is only to be used for OOM testing.
1.80 +*/
1.81 +void CWsMemoryManager::SetFailNextRetry()
1.82 + {
1.83 + iFailNextRetry = ETrue;
1.84 + }
1.85 +
1.86 +/**
1.87 +Implementing RAllocator
1.88 +*/
1.89 +
1.90 +/**
1.91 +Alloc and ReAlloc attempt to obtain memory through CWsTop::ReleaseMemory when they run low.
1.92 +ReleaseMemory looks for blocks of memory that the window server doesn't need urgently and frees
1.93 +them.
1.94 +*/
1.95 +TAny* CWsMemoryManager::Alloc(TInt aSize)
1.96 + {
1.97 + TBool keepTrying = ETrue;
1.98 + do
1.99 + {
1.100 + if(iReleasing)
1.101 + return iImpl->Alloc(aSize); //fallback on RAllocator
1.102 +
1.103 + if(TAny* ret = iImpl->Alloc(aSize)) //normal case
1.104 + return ret;
1.105 +
1.106 + TTime now;
1.107 + now.UniversalTime();
1.108 + if(now.MicroSecondsFrom(iCurrentBurstStart) < KBurstDuration)
1.109 + {
1.110 + iCurrentBurstReleaseCount++;
1.111 + if(iCurrentBurstReleaseCount > KMaxMemoryReleasesPerBurst)
1.112 + return NULL;
1.113 + }
1.114 + else
1.115 + {
1.116 + iCurrentBurstStart = now;
1.117 + iCurrentBurstReleaseCount = 1;
1.118 + }
1.119 +
1.120 + if(iReserveEnabled && iReserve && (aSize < iReserveSize))
1.121 + {
1.122 + Free(iReserve);
1.123 + iReserve = NULL;
1.124 + }
1.125 + else
1.126 + {
1.127 + iReleasing = ETrue;
1.128 + keepTrying = CWsTop::ReleaseMemory();
1.129 + if(keepTrying)
1.130 + {
1.131 + const TInt reclaimed = Compress(); //Try to give back to the OS
1.132 + }
1.133 + iReleasing = EFalse;
1.134 + }
1.135 +
1.136 + //used for OOM testing only
1.137 + if(iFailNextRetry)
1.138 + {
1.139 + iFailNextRetry = EFalse;
1.140 + keepTrying = EFalse;
1.141 + }
1.142 +
1.143 + } while(keepTrying);
1.144 +
1.145 + return NULL;
1.146 + }
1.147 +
1.148 +TAny* CWsMemoryManager::ReAlloc(TAny* aPtr, TInt aSize, TInt aMode)
1.149 + {
1.150 + TBool keepTrying = ETrue;
1.151 + do
1.152 + {
1.153 + if(iReleasing)
1.154 + return iImpl->ReAlloc(aPtr, aSize, aMode); //fallback on RAllocator
1.155 +
1.156 + if(TAny* ret = iImpl->ReAlloc(aPtr, aSize, aMode)) //normal case
1.157 + return ret;
1.158 +
1.159 + TTime now;
1.160 + now.UniversalTime();
1.161 + if(now.MicroSecondsFrom(iCurrentBurstStart) < KBurstDuration)
1.162 + {
1.163 + iCurrentBurstReleaseCount++;
1.164 + if(iCurrentBurstReleaseCount > KMaxMemoryReleasesPerBurst)
1.165 + return NULL;
1.166 + }
1.167 + else
1.168 + {
1.169 + iCurrentBurstStart = now;
1.170 + iCurrentBurstReleaseCount = 1;
1.171 + }
1.172 +
1.173 + if(iReserveEnabled && iReserve && (aSize < iReserveSize))
1.174 + {
1.175 + Free(iReserve);
1.176 + iReserve = NULL;
1.177 + }
1.178 + else
1.179 + {
1.180 + iReleasing = ETrue;
1.181 + keepTrying = CWsTop::ReleaseMemory();
1.182 + if(keepTrying)
1.183 + {
1.184 + const TInt reclaimed = Compress(); //Try to give back to the OS
1.185 + }
1.186 + iReleasing = EFalse;
1.187 + }
1.188 +
1.189 + //used for OOM testing only
1.190 + if(iFailNextRetry)
1.191 + {
1.192 + iFailNextRetry = EFalse;
1.193 + keepTrying = EFalse;
1.194 + }
1.195 +
1.196 + } while(keepTrying);
1.197 +
1.198 + return NULL;
1.199 + }
1.200 +
1.201 +/**
1.202 +The rest of these functions just call the default implementation
1.203 +*/
1.204 +void CWsMemoryManager::Free(TAny* aPtr)
1.205 + {
1.206 + return iImpl->Free(aPtr);
1.207 + }
1.208 +
1.209 +TInt CWsMemoryManager::AllocLen(const TAny* aCell) const
1.210 + {
1.211 + return iImpl->AllocLen(aCell);
1.212 + }
1.213 +
1.214 +TInt CWsMemoryManager::Compress()
1.215 + {
1.216 + return iImpl->Compress();
1.217 + }
1.218 +
1.219 +void CWsMemoryManager::Reset()
1.220 + {
1.221 + iImpl->Reset();
1.222 + }
1.223 +
1.224 +TInt CWsMemoryManager::AllocSize(TInt& aTotalAllocSize) const
1.225 + {
1.226 + return iImpl->AllocSize(aTotalAllocSize);
1.227 + }
1.228 +
1.229 +TInt CWsMemoryManager::Available(TInt& aBiggestBlock) const
1.230 + {
1.231 + return iImpl->Available(aBiggestBlock);
1.232 + }
1.233 +
1.234 +TInt CWsMemoryManager::DebugFunction(TInt aFunc, TAny* a1, TAny* a2)
1.235 + {
1.236 + return iImpl->DebugFunction(aFunc,a1,a2);
1.237 + }
1.238 +
1.239 +TInt CWsMemoryManager::Count() const
1.240 + {
1.241 + return iImpl->Count();
1.242 + }
1.243 +/** This is a fairly dumb way to enable and disable the reserve, but we normally
1.244 +get away with it because wserv is high priority. A better approach would be to
1.245 +use placement new into the reserve memory and manage it directly. This would also
1.246 +allow us to track misbehaving code which allocated during OOM drawing and didn't
1.247 +free at the end.
1.248 +*/
1.249 +void CWsMemoryManager::EnableReserve()
1.250 + {
1.251 + WS_ASSERT_DEBUG(!iReserveEnabled, EWsPanicMemoryManager);
1.252 + iReserveEnabled = ETrue;
1.253 + }
1.254 +
1.255 +void CWsMemoryManager::DisableReserve()
1.256 + {
1.257 + WS_ASSERT_DEBUG(iReserveEnabled, EWsPanicMemoryManager);
1.258 + iReserveEnabled = EFalse;
1.259 + if((!iReserve) && (iReserveSize > 0))
1.260 + iReserve = Alloc(iReserveSize);
1.261 + }