os/graphics/windowing/windowserver/nonnga/SERVER/WsMemMgr.cpp
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
Update contrib.
sl@0
     1
// Copyright (c) 2007-2009 Nokia Corporation and/or its subsidiary(-ies).
sl@0
     2
// All rights reserved.
sl@0
     3
// This component and the accompanying materials are made available
sl@0
     4
// under the terms of "Eclipse Public License v1.0"
sl@0
     5
// which accompanies this distribution, and is available
sl@0
     6
// at the URL "http://www.eclipse.org/legal/epl-v10.html".
sl@0
     7
//
sl@0
     8
// Initial Contributors:
sl@0
     9
// Nokia Corporation - initial contribution.
sl@0
    10
//
sl@0
    11
// Contributors:
sl@0
    12
//
sl@0
    13
// Description:
sl@0
    14
//
sl@0
    15
sl@0
    16
#include "WsMemMgr.h"
sl@0
    17
#include "inifile.h"
sl@0
    18
#include "panics.h"
sl@0
    19
#include "wstop.h"
sl@0
    20
sl@0
    21
CWsMemoryManager * CWsMemoryManager::iStatic = NULL;
sl@0
    22
sl@0
    23
CWsMemoryManager * CWsMemoryManager::Static()
sl@0
    24
	{
sl@0
    25
	return iStatic;
sl@0
    26
	}
sl@0
    27
sl@0
    28
CWsMemoryManager * CWsMemoryManager::NewLC()
sl@0
    29
	{
sl@0
    30
	CWsMemoryManager * self = new (ELeave) CWsMemoryManager;
sl@0
    31
	CleanupStack::PushL(self);
sl@0
    32
	self->ConstructL();
sl@0
    33
	iStatic = self;
sl@0
    34
	return iStatic;
sl@0
    35
	}
sl@0
    36
sl@0
    37
CWsMemoryManager::CWsMemoryManager()
sl@0
    38
	{
sl@0
    39
	iImpl = User::SwitchAllocator(this);
sl@0
    40
	}
sl@0
    41
	
sl@0
    42
CWsMemoryManager::~CWsMemoryManager()
sl@0
    43
	{
sl@0
    44
	WS_ASSERT_ALWAYS(this == User::SwitchAllocator(iImpl),EWsPanicMemoryManager);
sl@0
    45
	iStatic = 0;
sl@0
    46
	if (iReserve!=NULL)
sl@0
    47
		{
sl@0
    48
		Free(iReserve);
sl@0
    49
		iReserve = NULL;
sl@0
    50
		}	
sl@0
    51
	}
sl@0
    52
sl@0
    53
void CWsMemoryManager::ConstructL()
sl@0
    54
	{
sl@0
    55
	_LIT(KMemMgrReserve, "MEMORYRESERVE");
sl@0
    56
	const TInt KDefaultMemMgrReserve = 1024;
sl@0
    57
	
sl@0
    58
	if (!WsIniFile->FindVar(KMemMgrReserve, iReserveSize))
sl@0
    59
		iReserveSize = KDefaultMemMgrReserve;
sl@0
    60
sl@0
    61
	if (iReserveSize > 0)
sl@0
    62
		iReserve = Alloc(iReserveSize);
sl@0
    63
	}
sl@0
    64
sl@0
    65
/**
sl@0
    66
Implementing RAllocator
sl@0
    67
*/
sl@0
    68
sl@0
    69
/**
sl@0
    70
Alloc and ReAlloc attempt to obtain memory through CWsTop::ReleaseMemory when they run low.
sl@0
    71
ReleaseMemory looks for blocks of memory that the window server doesn't need urgently and frees
sl@0
    72
them.
sl@0
    73
*/
sl@0
    74
TAny* CWsMemoryManager::Alloc(TInt aSize)
sl@0
    75
	{
sl@0
    76
	TBool keepTrying = ETrue;
sl@0
    77
	do
sl@0
    78
		{
sl@0
    79
		if(iReleasing)
sl@0
    80
			return iImpl->Alloc(aSize); //fallback on RAllocator
sl@0
    81
		
sl@0
    82
		if(TAny* ret = iImpl->Alloc(aSize)) //normal case
sl@0
    83
			return ret;
sl@0
    84
		
sl@0
    85
		if(iReserveEnabled && iReserve && (aSize < iReserveSize))
sl@0
    86
			{
sl@0
    87
			Free(iReserve);
sl@0
    88
			iReserve = NULL;
sl@0
    89
			}
sl@0
    90
		else
sl@0
    91
			{
sl@0
    92
			iReleasing = ETrue;
sl@0
    93
			keepTrying = CWsTop::ReleaseMemory();
sl@0
    94
			if(keepTrying)
sl@0
    95
 				{
sl@0
    96
 				const TInt reclaimed = Compress(); //Try to give back to the OS
sl@0
    97
 				}			
sl@0
    98
			iReleasing = EFalse;
sl@0
    99
			}
sl@0
   100
		
sl@0
   101
		} while(keepTrying);
sl@0
   102
sl@0
   103
	return NULL;
sl@0
   104
	}
sl@0
   105
sl@0
   106
TAny* CWsMemoryManager::ReAlloc(TAny* aPtr, TInt aSize, TInt aMode)
sl@0
   107
	{
sl@0
   108
	TBool keepTrying = ETrue;
sl@0
   109
	do
sl@0
   110
		{
sl@0
   111
		if(iReleasing)
sl@0
   112
			return iImpl->ReAlloc(aPtr, aSize, aMode); //fallback on RAllocator
sl@0
   113
		
sl@0
   114
		if(TAny* ret = iImpl->ReAlloc(aPtr, aSize, aMode)) //normal case
sl@0
   115
			return ret;
sl@0
   116
		
sl@0
   117
		if(iReserveEnabled && iReserve && (aSize < iReserveSize))
sl@0
   118
			{
sl@0
   119
			Free(iReserve);
sl@0
   120
			iReserve = NULL;
sl@0
   121
			}
sl@0
   122
		else
sl@0
   123
			{
sl@0
   124
			iReleasing = ETrue;
sl@0
   125
			keepTrying = CWsTop::ReleaseMemory();
sl@0
   126
			if(keepTrying)
sl@0
   127
 				{
sl@0
   128
 				const TInt reclaimed = Compress(); //Try to give back to the OS
sl@0
   129
 				}	
sl@0
   130
			iReleasing = EFalse;
sl@0
   131
			}
sl@0
   132
sl@0
   133
		} while(keepTrying);
sl@0
   134
sl@0
   135
	return NULL;
sl@0
   136
	}
sl@0
   137
sl@0
   138
/**
sl@0
   139
The rest of these functions just call the default implementation
sl@0
   140
*/
sl@0
   141
void CWsMemoryManager::Free(TAny* aPtr)
sl@0
   142
	{
sl@0
   143
	return iImpl->Free(aPtr);
sl@0
   144
	}
sl@0
   145
sl@0
   146
TInt CWsMemoryManager::AllocLen(const TAny* aCell) const
sl@0
   147
	{
sl@0
   148
	return iImpl->AllocLen(aCell);
sl@0
   149
	}
sl@0
   150
sl@0
   151
TInt CWsMemoryManager::Compress()
sl@0
   152
	{
sl@0
   153
	return iImpl->Compress();
sl@0
   154
	}
sl@0
   155
sl@0
   156
void CWsMemoryManager::Reset()
sl@0
   157
	{
sl@0
   158
	iImpl->Reset();
sl@0
   159
	}
sl@0
   160
sl@0
   161
TInt CWsMemoryManager::AllocSize(TInt& aTotalAllocSize) const
sl@0
   162
	{
sl@0
   163
	return iImpl->AllocSize(aTotalAllocSize);
sl@0
   164
	}
sl@0
   165
sl@0
   166
TInt CWsMemoryManager::Available(TInt& aBiggestBlock) const
sl@0
   167
	{
sl@0
   168
	return iImpl->Available(aBiggestBlock);
sl@0
   169
	}
sl@0
   170
sl@0
   171
TInt CWsMemoryManager::DebugFunction(TInt aFunc, TAny* a1, TAny* a2)
sl@0
   172
	{
sl@0
   173
	return iImpl->DebugFunction(aFunc,a1,a2);
sl@0
   174
	}
sl@0
   175
sl@0
   176
TInt CWsMemoryManager::Count() const
sl@0
   177
	{
sl@0
   178
	return iImpl->Count();
sl@0
   179
	}
sl@0
   180
/** This is a fairly dumb way to enable and disable the reserve, but we normally
sl@0
   181
get away with it because wserv is high priority.  A better approach would be to
sl@0
   182
use placement new into the reserve memory and manage it directly.  This would also
sl@0
   183
allow us to track misbehaving code which allocated during OOM drawing and didn't
sl@0
   184
free at the end.
sl@0
   185
*/
sl@0
   186
void CWsMemoryManager::EnableReserve()
sl@0
   187
	{
sl@0
   188
	WS_ASSERT_DEBUG(!iReserveEnabled, EWsPanicMemoryManager);
sl@0
   189
	iReserveEnabled = ETrue;
sl@0
   190
	}
sl@0
   191
sl@0
   192
void CWsMemoryManager::DisableReserve()
sl@0
   193
	{
sl@0
   194
	WS_ASSERT_DEBUG(iReserveEnabled, EWsPanicMemoryManager);
sl@0
   195
	iReserveEnabled = EFalse;
sl@0
   196
	if((!iReserve) && (iReserveSize > 0))
sl@0
   197
		iReserve = Alloc(iReserveSize);
sl@0
   198
	}