os/persistentdata/persistentstorage/centralrepository/cenrepsrv/cachemgr.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) 2004-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 "srvrepos_noc.h"
sl@0
    17
#include "cachemgr.h"
sl@0
    18
#include <bsul/bsul.h>
sl@0
    19
sl@0
    20
#define UNUSED_VAR(a) a = a
sl@0
    21
sl@0
    22
_LIT(KCacheLit, "CoarseGrainedCache");
sl@0
    23
_LIT(KDefaultCacheSizeLit, "size");
sl@0
    24
_LIT(KDefaultEvictionTimeoutLit, "timeout");
sl@0
    25
    
sl@0
    26
CRepositoryCacheManager* CRepositoryCacheManager::NewLC(RFs& aFs)
sl@0
    27
	{
sl@0
    28
	CRepositoryCacheManager* self = new(ELeave) CRepositoryCacheManager;
sl@0
    29
	CleanupStack::PushL(self);
sl@0
    30
	self->ConstructL(aFs);
sl@0
    31
	return self;
sl@0
    32
	}
sl@0
    33
sl@0
    34
CRepositoryCacheManager::~CRepositoryCacheManager()
sl@0
    35
	{
sl@0
    36
	DisableCache(ETrue);
sl@0
    37
	}
sl@0
    38
sl@0
    39
void CRepositoryCacheManager::ConstructL(RFs& aFs)
sl@0
    40
	{
sl@0
    41
	CTimer::ConstructL();
sl@0
    42
	
sl@0
    43
	BSUL::CIniFile16* iniFile = NULL;
sl@0
    44
	TInt res = KErrNone;
sl@0
    45
	TBuf<KMaxFileName> iniFileName;
sl@0
    46
	
sl@0
    47
	iniFileName.Copy( *TServerResources::iInstallDirectory );
sl@0
    48
	iniFileName.Append( KCacheMgrIniFile );	
sl@0
    49
	TRAP(res, iniFile = BSUL::CIniFile16::NewL(aFs, iniFileName, ETrue));
sl@0
    50
	if(res==KErrNotFound)
sl@0
    51
		{
sl@0
    52
		// if RomDirectory exists
sl@0
    53
		if (TServerResources::iRomDirectory)
sl@0
    54
			{
sl@0
    55
			iniFileName.Copy( *TServerResources::iRomDirectory );
sl@0
    56
			iniFileName.Append( KCacheMgrIniFile );	
sl@0
    57
			TRAP(res, iniFile = BSUL::CIniFile16::NewL(aFs, iniFileName, ETrue));
sl@0
    58
			}
sl@0
    59
		if(res==KErrNotFound)
sl@0
    60
			{
sl@0
    61
			__CENTREP_TRACE1("CENTREP: Central Repository Cache Parameters ini file %S not found. Default values will be used.", &KCacheMgrIniFile);
sl@0
    62
			return;
sl@0
    63
			}
sl@0
    64
		}
sl@0
    65
	if (res != KErrNone)
sl@0
    66
		{
sl@0
    67
		User::Leave(res);
sl@0
    68
		}
sl@0
    69
		
sl@0
    70
	CleanupStack::PushL(iniFile);
sl@0
    71
	
sl@0
    72
	TBuf<20> buffer;
sl@0
    73
	TPtrC ptr(buffer);
sl@0
    74
	
sl@0
    75
	// find the value
sl@0
    76
	res = iniFile->FindVar(KCacheLit(), KDefaultCacheSizeLit(), ptr);
sl@0
    77
	TLex lex(ptr);
sl@0
    78
sl@0
    79
	TBool valueFound = EFalse;
sl@0
    80
	
sl@0
    81
	// if the value can't be found or can't be converted into a positive integer, use the default
sl@0
    82
	if (res != KErrNone || lex.Val(iCacheSize) != KErrNone || iCacheSize < 0)	
sl@0
    83
		{
sl@0
    84
		iCacheSize = KDefaultCacheSize;
sl@0
    85
		}
sl@0
    86
	else
sl@0
    87
		{
sl@0
    88
		valueFound = ETrue;			
sl@0
    89
		}
sl@0
    90
		
sl@0
    91
	// if the value can be found, convert it
sl@0
    92
	if (iniFile->FindVar(KCacheLit(), KDefaultEvictionTimeoutLit(), ptr) == KErrNone)
sl@0
    93
		{
sl@0
    94
		TInt tempTimeout;
sl@0
    95
		lex.Assign(ptr);
sl@0
    96
		// if the value can be converted into a positive integer, assign it to iDefaultTimeout.
sl@0
    97
		if (lex.Val(tempTimeout) == KErrNone && tempTimeout >= 0)
sl@0
    98
			{
sl@0
    99
			iDefaultTimeout = tempTimeout;
sl@0
   100
			valueFound = ETrue;
sl@0
   101
			}
sl@0
   102
		}
sl@0
   103
	
sl@0
   104
#ifdef _DEBUG
sl@0
   105
	// in Debug mode, if the Cache ini file exists either in install directory or 
sl@0
   106
	// rom directory but does not contains the correct section "CoarseGrainedCache"
sl@0
   107
	// nor any key of "size" and "timeout", the server panics.
sl@0
   108
	if(! valueFound)
sl@0
   109
	{
sl@0
   110
	Panic(ECacheIniFileCorrupted);
sl@0
   111
	}
sl@0
   112
#else
sl@0
   113
	UNUSED_VAR(valueFound);
sl@0
   114
#endif		
sl@0
   115
sl@0
   116
	CleanupStack::PopAndDestroy(iniFile);
sl@0
   117
	}
sl@0
   118
sl@0
   119
void CRepositoryCacheManager::EnableCache(TInt aDefaultTimeout, TInt aCacheSize)
sl@0
   120
	{
sl@0
   121
	if (aDefaultTimeout>0)
sl@0
   122
		{
sl@0
   123
		iDefaultTimeout = aDefaultTimeout;
sl@0
   124
		}
sl@0
   125
	if (aCacheSize>0)
sl@0
   126
		{
sl@0
   127
		iCacheSize = aCacheSize;
sl@0
   128
		}
sl@0
   129
	
sl@0
   130
	EnableCache();
sl@0
   131
	}
sl@0
   132
sl@0
   133
void CRepositoryCacheManager::EnableCache()
sl@0
   134
	{
sl@0
   135
	// If disabled, enable
sl@0
   136
	if (!iEnabled)
sl@0
   137
		{
sl@0
   138
		iEnabled = ETrue;
sl@0
   139
		__CENTREP_TRACE2("CENTREP: Cache Enabled. Size:%d Default Timeout:%d", iCacheSize, iDefaultTimeout.Int());
sl@0
   140
		}
sl@0
   141
	}
sl@0
   142
		
sl@0
   143
void CRepositoryCacheManager::DisableCache(TBool aFullFlush)
sl@0
   144
	{
sl@0
   145
	// If enabled, disable
sl@0
   146
	if (iEnabled)
sl@0
   147
		{
sl@0
   148
		// cancel any outstanding timer
sl@0
   149
		Cancel();
sl@0
   150
sl@0
   151
		FlushCache(aFullFlush);
sl@0
   152
sl@0
   153
		iEnabled = EFalse;
sl@0
   154
		__CENTREP_TRACE("CENTREP: Cache Disabled");
sl@0
   155
		}
sl@0
   156
	}
sl@0
   157
sl@0
   158
void CRepositoryCacheManager::RescheduleTimer(const TTime& aTimeInUTC)
sl@0
   159
	{
sl@0
   160
	
sl@0
   161
	TTime now;
sl@0
   162
	now.UniversalTime();
sl@0
   163
	
sl@0
   164
	//Get the 64bit time interval between now and the cache timeout
sl@0
   165
	TTimeIntervalMicroSeconds interval64 = aTimeInUTC.MicroSecondsFrom(now);
sl@0
   166
	TTimeIntervalMicroSeconds32 interval32(iDefaultTimeout);
sl@0
   167
	
sl@0
   168
	//If the interval is positive, i.e. the timeout is in the future, convert 
sl@0
   169
	//this interval to a 32 bit value, otherwise use the default timeout
sl@0
   170
	if(interval64 > 0)
sl@0
   171
		{
sl@0
   172
		//If the interval value is less than the maximum 32 bit value cast
sl@0
   173
		//interval to 32 bit value, otherwise the interval is too large for 
sl@0
   174
		//a 32 bit value so just set the interval to the max 32 bit value
sl@0
   175
		const TInt64 KMax32BitValue(KMaxTInt32);
sl@0
   176
		interval32 = (interval64 <= KMax32BitValue) ? 
sl@0
   177
				static_cast<TTimeIntervalMicroSeconds32>(interval64.Int64()): KMaxTInt32;
sl@0
   178
		}
sl@0
   179
sl@0
   180
	//Reschedule the timer
sl@0
   181
	After(interval32);
sl@0
   182
sl@0
   183
	}
sl@0
   184
sl@0
   185
void CRepositoryCacheManager::RemoveIdleRepository(CSharedRepository* aRepository)
sl@0
   186
	{
sl@0
   187
	if (iEnabled)
sl@0
   188
		{
sl@0
   189
		TInt i;
sl@0
   190
		TInt count=iIdleRepositories.Count();
sl@0
   191
		for(i=count-1; i>=0; i--)
sl@0
   192
			{
sl@0
   193
			if(iIdleRepositories[i].iSharedRepository==aRepository)
sl@0
   194
				{
sl@0
   195
				break;
sl@0
   196
				}
sl@0
   197
			}
sl@0
   198
		
sl@0
   199
		// Idle repository might not be found in the list if multiple clients try to open the same 
sl@0
   200
		// repository at the same time. First client will remove it and second one will not find it
sl@0
   201
		if(i>=0)
sl@0
   202
			{
sl@0
   203
			__CENTREP_TRACE1("CENTREP: Cache Hit when opening repository %x", aRepository->Uid().iUid);
sl@0
   204
sl@0
   205
			iTotalCacheUsage -= iIdleRepositories[i].iSharedRepository->Size();		
sl@0
   206
			iIdleRepositories.Remove(i);
sl@0
   207
			
sl@0
   208
			// If this was the first repository on the list, it means its timer is still ticking. 
sl@0
   209
			// We have to stop it and ...
sl@0
   210
			if (i==0)
sl@0
   211
				{
sl@0
   212
				Cancel();
sl@0
   213
				// if there's still other repositories in the list, reschedule the timer with the
sl@0
   214
				// new top-of-the-list  
sl@0
   215
				if (iIdleRepositories.Count())
sl@0
   216
					{
sl@0
   217
					RescheduleTimer(iIdleRepositories[0].iCacheTime);
sl@0
   218
					}
sl@0
   219
				}
sl@0
   220
			}
sl@0
   221
		else
sl@0
   222
			{
sl@0
   223
			__CENTREP_TRACE1("CENTREP: Cache Miss when opening repository %x", aRepository->Uid().iUid);
sl@0
   224
			}
sl@0
   225
		}
sl@0
   226
	}
sl@0
   227
sl@0
   228
#ifdef CACHE_OOM_TESTABILITY
sl@0
   229
  	// This code is only for tesing and doesn't go into MCL
sl@0
   230
	// Hence hide the leave in a macro instead of making StartEvictionL
sl@0
   231
#define TEST_CODE_LEAVE(x) User::Leave(x)
sl@0
   232
#endif	
sl@0
   233
sl@0
   234
TBool CRepositoryCacheManager::StartEviction(CSharedRepository*& aRepository)
sl@0
   235
	{
sl@0
   236
	// find the item in the cache and remove it if it exists to reset the timer
sl@0
   237
	RemoveIdleRepository(aRepository);
sl@0
   238
sl@0
   239
	TInt64 lastTop = 0;
sl@0
   240
	
sl@0
   241
	if (iIdleRepositories.Count())
sl@0
   242
		{
sl@0
   243
		lastTop = iIdleRepositories[0].iCacheTime.Int64();
sl@0
   244
		}
sl@0
   245
sl@0
   246
	// Execute the forced eviction algorithm only if it will make sense
sl@0
   247
	// The eviction makes sense if:
sl@0
   248
	// - there's anything in the cache to evict
sl@0
   249
	// - the repository we're trying to cache can fit in the cache after evictions
sl@0
   250
	if (iIdleRepositories.Count() && (aRepository->Size() < iCacheSize))
sl@0
   251
		{
sl@0
   252
		// Check to see if current cache size + the current repository size is overshooting the limit
sl@0
   253
		if (iTotalCacheUsage + aRepository->Size() > iCacheSize)
sl@0
   254
			{
sl@0
   255
			// Forced eviction
sl@0
   256
			__CENTREP_TRACE3("CENTREP: Cache Size Exceeded: Current(%d)+Size(%d)>Cache(%d)", iTotalCacheUsage, aRepository->Size(), iCacheSize);
sl@0
   257
			
sl@0
   258
			// Sort in the forced eviction order
sl@0
   259
			TLinearOrder<TRepositoryCacheInfo> forcedSortOrder(CRepositoryCacheManager::ForcedEvictionSortOrder);
sl@0
   260
			iIdleRepositories.Sort(forcedSortOrder);
sl@0
   261
			
sl@0
   262
			// Evict one by one until there's enough cache space or we run out of idle reps
sl@0
   263
			do
sl@0
   264
				{
sl@0
   265
				__CENTREP_TRACE1("CENTREP: Forced Eviction of repository %x", iIdleRepositories[0].iSharedRepository->Uid().iUid);			
sl@0
   266
				iTotalCacheUsage -= iIdleRepositories[0].iSharedRepository->Size();
sl@0
   267
				Evict(0);
sl@0
   268
				iIdleRepositories.Remove(0);		
sl@0
   269
				} while (iIdleRepositories.Count() && (iTotalCacheUsage + aRepository->Size() > iCacheSize));
sl@0
   270
			
sl@0
   271
#ifdef CENTREP_TRACE			
sl@0
   272
			if (!iIdleRepositories.Count())
sl@0
   273
				{
sl@0
   274
				__CENTREP_TRACE("CENREP: Cache Emptied by Forced Eviction");
sl@0
   275
				}
sl@0
   276
#endif				
sl@0
   277
			// Re-sort to timer order for normal operation
sl@0
   278
			TLinearOrder<TRepositoryCacheInfo> timerSortOrder(CRepositoryCacheManager::TimerEvictionSortOrder);
sl@0
   279
			iIdleRepositories.Sort(timerSortOrder);
sl@0
   280
			};
sl@0
   281
		}
sl@0
   282
	
sl@0
   283
	// See if there's enough space now
sl@0
   284
	if (iTotalCacheUsage + aRepository->Size() > iCacheSize)
sl@0
   285
		{
sl@0
   286
		return EFalse;
sl@0
   287
		}
sl@0
   288
sl@0
   289
	// Create new item for the cache and insert it in the list
sl@0
   290
	TRepositoryCacheInfo repInfo;
sl@0
   291
	
sl@0
   292
	repInfo.iCacheTime.UniversalTime();
sl@0
   293
	repInfo.iCacheTime += TTimeIntervalMicroSeconds32(iDefaultTimeout);
sl@0
   294
	repInfo.iSharedRepository = aRepository;
sl@0
   295
	
sl@0
   296
	TLinearOrder<TRepositoryCacheInfo> timerSortOrder(CRepositoryCacheManager::TimerEvictionSortOrder);
sl@0
   297
	// With the same timeout value assigned to all repositories, no two repositories can have the same 
sl@0
   298
	// timeout theoretically, so InsertInOrder is sufficient. But in practice, because of the poor 
sl@0
   299
	// resolution of the NTickCount() function used by TTime::UniversalTime(), InsertInOrderAllowRepeats 
sl@0
   300
	// should be used instead of InsertInOrder to allow for duplicate timer values caused by two 
sl@0
   301
	// repositories cached in quick succession (<1ms)
sl@0
   302
	TInt err = iIdleRepositories.InsertInOrderAllowRepeats(repInfo, timerSortOrder);
sl@0
   303
#ifdef CACHE_OOM_TESTABILITY
sl@0
   304
  	// This code is only for tesing and doesn't go into MCL
sl@0
   305
  	if (err == KErrNoMemory)	
sl@0
   306
  		{
sl@0
   307
  		TServerResources::iObserver->RemoveOpenRepository(aRepository);
sl@0
   308
  		aRepository = NULL;
sl@0
   309
  		// Should Leave here for the OOM tests to successfully complete. 
sl@0
   310
		TEST_CODE_LEAVE(err);
sl@0
   311
  		}
sl@0
   312
#endif	
sl@0
   313
	if (err!=KErrNone)
sl@0
   314
		{
sl@0
   315
		return EFalse;
sl@0
   316
		}
sl@0
   317
sl@0
   318
	iTotalCacheUsage += repInfo.iSharedRepository->Size();
sl@0
   319
	
sl@0
   320
	// Only reset timer if necessary. This operation takes time and doing it every time reduces performance considerably
sl@0
   321
	if (lastTop != iIdleRepositories[0].iCacheTime.Int64())
sl@0
   322
		{
sl@0
   323
		// reset timer to the new top-of-the-list
sl@0
   324
		Cancel();
sl@0
   325
		RescheduleTimer(iIdleRepositories[0].iCacheTime);
sl@0
   326
		}
sl@0
   327
		
sl@0
   328
	return ETrue;
sl@0
   329
	}
sl@0
   330
sl@0
   331
void CRepositoryCacheManager::FlushCache(TBool aFullFlush)
sl@0
   332
	{
sl@0
   333
	// iterate through idle repositories (loaded in memory, scheduled to be evicted)
sl@0
   334
	TInt idleRepCount = iIdleRepositories.Count();
sl@0
   335
	for(TInt repCount = idleRepCount - 1; repCount >= 0 ; repCount--)	
sl@0
   336
		{
sl@0
   337
		// check if there are any observers listening (to see if any client is connected to this repository)
sl@0
   338
		if (aFullFlush || (TServerResources::iObserver->FindConnectedRepository(iIdleRepositories[repCount].iSharedRepository->Uid())==KErrNotFound))
sl@0
   339
			{
sl@0
   340
			// if the client has already been disconnected, unload from memory
sl@0
   341
			Evict(repCount);
sl@0
   342
			}
sl@0
   343
		}
sl@0
   344
	// this whole iteration and search above can be replaced by a simple reference counter variable check,
sl@0
   345
	// if the server is redesigned using a resource manager type pattern with CSharedRepository object as a resource
sl@0
   346
	
sl@0
   347
	// empty the list
sl@0
   348
	iIdleRepositories.Reset();
sl@0
   349
	
sl@0
   350
	iTotalCacheUsage = 0;
sl@0
   351
	__CENTREP_TRACE1("CENTREP: Cache Flush: %d repositories flushed", idleRepCount);
sl@0
   352
	}
sl@0
   353
	
sl@0
   354
// Evict removes items from iOpenRepositories list, not from iIdleRepositories list
sl@0
   355
void CRepositoryCacheManager::Evict(TInt aIdleRepIndex)
sl@0
   356
	{
sl@0
   357
	// find,remove and delete the idle repositories' pointers in the open repositories list 
sl@0
   358
	TServerResources::iObserver->RemoveOpenRepository(iIdleRepositories[aIdleRepIndex].iSharedRepository);
sl@0
   359
	}
sl@0
   360
		
sl@0
   361
void CRepositoryCacheManager::RunL()
sl@0
   362
	{
sl@0
   363
	TTime now;
sl@0
   364
sl@0
   365
	now.UniversalTime();
sl@0
   366
sl@0
   367
	TInt count = iIdleRepositories.Count();
sl@0
   368
	
sl@0
   369
	// repositories that are involved in active transactions are not idle.
sl@0
   370
	// this checks to make sure that we're not trying to reclaim memory that
sl@0
   371
	// is actually still currently in use.
sl@0
   372
	
sl@0
   373
	for (TInt i = 0;i < count;i++)
sl@0
   374
		{
sl@0
   375
		if (iIdleRepositories[i].iCacheTime > now)
sl@0
   376
			{
sl@0
   377
			break;
sl@0
   378
			}
sl@0
   379
		
sl@0
   380
		if (iIdleRepositories[i].iSharedRepository->IsTransactionActive())
sl@0
   381
			{
sl@0
   382
			__CENTREP_TRACE1("CRepositoryCacheManager::RunL - rescheduling UID 0x%x, in active transaction",
sl@0
   383
					iIdleRepositories[i].iSharedRepository->Uid().iUid);
sl@0
   384
			StartEviction(iIdleRepositories[i].iSharedRepository);
sl@0
   385
			return;
sl@0
   386
			}
sl@0
   387
		}
sl@0
   388
	
sl@0
   389
sl@0
   390
	// Try to evict all the repositories which have expired. There might be more than one repository
sl@0
   391
	// destined to expire at the same time, or there are more than one repository with expiry times
sl@0
   392
	// between the scheduled expiry time and now (which theoretically should have been the same, but maybe
sl@0
   393
	// because of other procesor activity, the timer Active Object just got late a bit)
sl@0
   394
	while((iIdleRepositories.Count()) && (iIdleRepositories[0].iCacheTime<=now))
sl@0
   395
		{
sl@0
   396
		__CENTREP_TRACE1("CENTREP: Normal Eviction of repository %x", iIdleRepositories[0].iSharedRepository->Uid().iUid);			
sl@0
   397
		// Always remove from the top of the sorted list
sl@0
   398
		iTotalCacheUsage -= iIdleRepositories[0].iSharedRepository->Size();		
sl@0
   399
		Evict(0);
sl@0
   400
		iIdleRepositories.Remove(0);		
sl@0
   401
		};
sl@0
   402
		
sl@0
   403
	// reschedule to run again at the expiry date of next repository on the list, if any
sl@0
   404
	if (iIdleRepositories.Count())
sl@0
   405
		{
sl@0
   406
		RescheduleTimer(iIdleRepositories[0].iCacheTime);
sl@0
   407
		}
sl@0
   408
	else
sl@0
   409
		{
sl@0
   410
		__CENTREP_TRACE("CENTREP: Cache Empty/Timer Disabled");			
sl@0
   411
		}
sl@0
   412
	}
sl@0
   413
sl@0
   414
TInt CRepositoryCacheManager::ForcedEvictionSortOrder(const TRepositoryCacheInfo &aRepository1, const TRepositoryCacheInfo &aRepository2)
sl@0
   415
	{
sl@0
   416
/*
sl@0
   417
   The code in the comments below is the original simple-to-read version of the algebraically
sl@0
   418
   simplified production code. 
sl@0
   419
sl@0
   420
	TTime now;
sl@0
   421
sl@0
   422
	now.UniversalTime();
sl@0
   423
sl@0
   424
	// we calculate the ages of the repositories by taking the difference between now and when
sl@0
   425
	// they were last became idle. When refactoring, the calculation of the absolute ages will be 
sl@0
   426
	// eleminated and the age difference between the repositories will be used in the formula instead
sl@0
   427
	
sl@0
   428
	TTimeIntervalMicroSeconds age1 = now.MicroSecondsFrom(aRepository1.iCacheTime);
sl@0
   429
	TTimeIntervalMicroSeconds age2 = now.MicroSecondsFrom(aRepository2.iCacheTime);
sl@0
   430
	
sl@0
   431
	// then divide the resulting numbers by conversion constant to get a number in a compatible unit
sl@0
   432
	// to the size. This operation reduces the microsecond-based values to having an approx. max
sl@0
   433
	// of 100K units
sl@0
   434
sl@0
   435
	TInt t1 = age1.Int64()/KTimeoutToSizeConversion;
sl@0
   436
	TInt t2 = age2.Int64()/KTimeoutToSizeConversion;
sl@0
   437
	
sl@0
   438
	// the resulting normalized time difference values are added to the size of the repository
sl@0
   439
	// resulting in an implicit %50 weight in the overall importance value 
sl@0
   440
	// An approx. maximum size of a repository is assumed to be around 100K
sl@0
   441
	
sl@0
   442
	TInt importance1 = t1+aRepository1.iSharedRepository->Size();
sl@0
   443
	TInt importance2 = t2+aRepository2.iSharedRepository->Size();
sl@0
   444
	
sl@0
   445
	// the difference between the importances of the repositories determine the sorting order
sl@0
   446
sl@0
   447
	return static_cast<TInt>(importance1-importance2);
sl@0
   448
*/	
sl@0
   449
	//	after refactoring, the resulting formula is this one
sl@0
   450
	return static_cast<TInt>(((aRepository1.iCacheTime.Int64()-aRepository2.iCacheTime.Int64())/KTimeoutToSizeConversion)+(aRepository1.iSharedRepository->Size()-aRepository2.iSharedRepository->Size()));	
sl@0
   451
	}
sl@0
   452
sl@0
   453
TInt CRepositoryCacheManager::TimerEvictionSortOrder(const TRepositoryCacheInfo &aRepository1, const TRepositoryCacheInfo &aRepository2)
sl@0
   454
	{
sl@0
   455
	return static_cast<TInt>(aRepository1.iCacheTime.Int64()-aRepository2.iCacheTime.Int64());
sl@0
   456
	}