os/persistentdata/persistentstorage/centralrepository/test/t_cenrep_cache.cpp
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
Update contrib.
     1 // Copyright (c) 2005-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".
     7 //
     8 // Initial Contributors:
     9 // Nokia Corporation - initial contribution.
    10 //
    11 // Contributors:
    12 //
    13 // Description:
    14 //
    15 
    16 #include <e32test.h>
    17 #include <f32file.h>
    18 #include <badesca.h>
    19 
    20 #include "srvrepos_noc.h"
    21 #include "srvres.h"
    22 #include "cachemgr.h"
    23 #include "sessnotf.h"
    24 
    25 RTest TheTest(_L("Central Repository Coarse-Grained Cache Tests"));
    26 
    27 enum TBorderTestStage
    28 	{
    29 	ENominalNoFile = -1,
    30 	ENominal = 0,
    31 	ESizeMin,
    32 	ESizeMinplus,
    33 	ESizeMaxminus,
    34 	ESizeMax,
    35 	ESizeMaxplus,
    36 	ETimeoutMin,
    37 	ETimeoutMinplus,
    38 	ETimeoutMaxminus,
    39 	EReallyWorstCase,
    40 	ELastStage
    41 	};
    42 
    43 const TUid KUidCacheTestRepositorySm	= { 0x00000002 };
    44 const TUid KUidCacheTestRepositorySm2	= { 0x00000011 };
    45 const TUid KUidCacheTestRepositoryMed   = { 0x11111111 };
    46 const TUid KUidCacheTestRepositoryMed2  = { 0x00057778 };
    47 const TUid KUidCacheTestRepositoryLrg	= { 0xCCCCCC01 };
    48 
    49 const TInt KTimerDelay = 1000000;
    50 const TInt KMemoryFiller = 3762;
    51 
    52 _LIT(KCacheMgrIniFileFolder, "\\private\\10202BE9\\");
    53 _LIT(KCacheMgrIniSrcFile, "centrepcache.ini");
    54 
    55 TTime starttime;
    56 
    57 class TRepositoryCacheManagerTester
    58 	{
    59 	public:
    60 	TRepositoryCacheManagerTester(): iOOMTest(EFalse), iStage(ENominalNoFile), iTestStepStage(0)
    61 		{
    62 		iFs.Connect();
    63 		}
    64 	~TRepositoryCacheManagerTester()
    65 		{
    66 		iFs.Close();
    67 		}
    68 	TInt CacheRepositoryL(TUid aRepUid, TBool& aTrap);
    69 	TInt CacheRepositoryL(TUid aRepUid);
    70 	TBool FindRepository(TUid aRepUid);
    71 	void FuncTestsL();
    72 	void SizeTestsL();
    73 	void TimingTestsL();
    74 	void DefectTestsL();
    75 	void DEF093491L();
    76 	void INC105967();
    77 	void DeleteFilesL();
    78 	void DEF111734L();
    79 	void DEF124147();
    80 	void InstallIniFileL( TBorderTestStage aFileSet );
    81 	void AdvanceToStageL(TBorderTestStage aStage);
    82 	void NextTest(TPtrC aMsg);
    83 	static TInt Callback(TAny* aParent);
    84 	static void CleanUp(TAny*);
    85 public:
    86 	TBool iOOMTest;
    87 	RFs iFs;
    88 	TBorderTestStage iStage;
    89 	TInt iTestStepStage;
    90 	};
    91 
    92 LOCAL_C TRepositoryCacheManagerTester Tester;
    93 
    94 LOCAL_C TCleanupItem gCleanup(TRepositoryCacheManagerTester::CleanUp);
    95 ///////////////////////////////////////////////////////////////////////////////////////
    96 ///////////////////////////////////////////////////////////////////////////////////////
    97 //Test macroses and functions
    98 
    99 void TRepositoryCacheManagerTester::DeleteFilesL()
   100 	{
   101 	_LIT( KOldInstallFiles, "c:\\private\\10202BE9\\*.ini" );
   102 
   103 	CFileMan* fm = CFileMan::NewL( iFs );
   104 	CleanupStack::PushL( fm );
   105 
   106 	TInt r = fm->Delete( KOldInstallFiles );
   107 	if ( r != KErrNone && r != KErrNotFound && r != KErrPathNotFound )
   108 		User::Leave(r);
   109 
   110 	CleanupStack::PopAndDestroy( fm );
   111 	}
   112 
   113 LOCAL_C void Check( TInt aValue, TInt aLine )
   114 	{
   115 	if ( !aValue )
   116 		{
   117 		TheTest( EFalse, aLine );
   118 		}
   119 	}
   120 
   121 LOCAL_C void Check( TInt aValue, TInt aExpected, TInt aLine )
   122 	{
   123 	if ( aValue != aExpected )
   124 		{
   125 		RDebug::Print( _L( "*** Expected error: %d, got: %d\r\n"), aExpected, aValue );
   126 		TheTest( EFalse, aLine );
   127 		}
   128 	}
   129 
   130 #define TEST(arg) ::Check((arg), __LINE__)
   131 #define TEST2(aValue, aExpected) ::Check(aValue, aExpected, __LINE__)
   132 
   133 ///////////////////////////////////////////////////////////////////////////////////////
   134 ///////////////////////////////////////////////////////////////////////////////////////
   135 
   136 void TRepositoryCacheManagerTester::NextTest(TPtrC aMsg)
   137 	{
   138 	if (!iOOMTest)
   139 		{
   140 		TheTest.Next( aMsg );
   141 		}
   142 	}
   143 
   144 void TRepositoryCacheManagerTester::AdvanceToStageL(TBorderTestStage aStage)
   145 	{
   146 	iStage = aStage;
   147 	if (aStage != ENominalNoFile)
   148 		{
   149 		InstallIniFileL( aStage );
   150 		}
   151 	else
   152 		{
   153 		DeleteFilesL();
   154 		}
   155 	delete TServerResources::iCacheManager;
   156 	TServerResources::iCacheManager = NULL;
   157 	TServerResources::iCacheManager = CRepositoryCacheManager::NewLC(iFs);
   158 	CleanupStack::Pop(TServerResources::iCacheManager);
   159 	}
   160 
   161 void TRepositoryCacheManagerTester::InstallIniFileL( TBorderTestStage aFileSet )
   162 	{
   163 	_LIT( KDriveC, "c:" );
   164 	_LIT( KDriveZ, "z:" );
   165 
   166 	TBuf<KMaxFileName> src1;
   167 	TBuf<KMaxFileName> dest1;
   168 	TInt r;
   169 
   170 	DeleteFilesL();
   171 
   172 	CFileMan* fm = CFileMan::NewL( iFs );
   173 	CleanupStack::PushL( fm );
   174 
   175 	if ((aFileSet>=ELastStage)||(aFileSet<0))
   176 		{
   177 		RDebug::Print( _L( "Illegal parameter to function: %d\r\n" ), aFileSet );
   178 		TheTest( EFalse, __LINE__ );
   179 		}
   180 	else
   181 		{
   182 		dest1.Copy( KDriveC );
   183 		dest1.Append( KCacheMgrIniFileFolder );
   184 		dest1.Append( KCacheMgrIniFile );
   185 		r = fm->Delete( dest1 );
   186 		if ( r != KErrNone && r != KErrNotFound && r != KErrPathNotFound )
   187 			User::Leave( r );
   188 		r = iFs.MkDirAll( dest1 );
   189 		if ( r != KErrNone && r != KErrAlreadyExists )
   190 			User::Leave( r );
   191 		src1.Copy( KDriveZ );
   192 		src1.Append( KCacheMgrIniFileFolder );
   193 		src1.Append( KCacheMgrIniSrcFile );
   194 		TBuf<2> testNo;
   195 		testNo.Num(aFileSet);
   196 		src1.Append( testNo );
   197 		User::LeaveIfError( fm->Copy( src1, dest1 ) );
   198 		r = fm->Attribs( dest1, KEntryAttArchive, KEntryAttReadOnly, TTime( 0 ), CFileMan::ERecurse );
   199 		TEST2( r, KErrNone );
   200 		}
   201 
   202 	CleanupStack::PopAndDestroy( fm );
   203 	}
   204 
   205 TInt TRepositoryCacheManagerTester::CacheRepositoryL(TUid aRepUid)
   206 	{
   207 	TBool trap = EFalse;
   208 	return CacheRepositoryL(aRepUid, trap);
   209 	}
   210 
   211 void TRepositoryCacheManagerTester::CleanUp(TAny*)
   212 	{
   213 	// If cache manager is initialized and used before, we flush it
   214 	if (TServerResources::iCacheManager)
   215 		{
   216 		TServerResources::iCacheManager->FlushCache();
   217 		}
   218 	// To get rid of the iOpenRepositories array leaking problem during OOM testing.
   219 	TServerResources::iObserver->CloseiOpenRepositories();
   220 	TServerResources::iObserver->Reset();
   221 	// To get rid of the array leaking problems during OOM testing.
   222 	TServerResources::iOwnerIdLookUpTable.Reset();
   223 	}
   224 
   225 // Opens and Closes a repository to place it in the cache
   226 TInt TRepositoryCacheManagerTester::CacheRepositoryL(TUid aRepUid, TBool& aTrap)
   227 	{
   228 	TInt repsize = 0;
   229 	// Notifier needed to open repositories.
   230 	CSessionNotifier* notifier;
   231 	notifier = new(ELeave)CSessionNotifier;
   232 	CleanupStack::PushL(notifier);
   233 
   234     CServerRepository* repository = new(ELeave) CServerRepository();
   235     CleanupStack::PushL(repository);
   236     TServerResources::iObserver->iTrapOOMOnOpen = aTrap;
   237     aTrap = EFalse; // This means that this function didn't leave until attempting to open the repository
   238     repository->OpenL(aRepUid,*notifier);
   239     repsize = repository->SizeiRepository();
   240 	repository->Close();
   241 	TServerResources::iObserver->iTrapOOMOnOpen = EFalse;
   242 
   243    	CleanupStack::PopAndDestroy(repository);
   244    	CleanupStack::PopAndDestroy(notifier);
   245 
   246    	return repsize;
   247 	}
   248 
   249 // Checks if a repository is in the cache
   250 TBool TRepositoryCacheManagerTester::FindRepository(TUid aRepUid)
   251 	{
   252 	for(TInt i=TServerResources::iCacheManager->iIdleRepositories.Count()-1; i>=0; i--)
   253 		{
   254 		if(TServerResources::iCacheManager->iIdleRepositories[i].iSharedRepository->Uid()==aRepUid)
   255 			{
   256 			return ETrue;
   257 			}
   258 		}
   259 	return EFalse;
   260 	}
   261 
   262 TInt TRepositoryCacheManagerTester::Callback(TAny* aParent)
   263 	{
   264 	TTime now;
   265 	now.UniversalTime();
   266 	TheTest.Printf(_L("Timer Expired at %Ld\n"), now.MicroSecondsFrom(starttime).Int64());
   267 	((CPeriodic*)aParent)->Cancel();;
   268 	CActiveScheduler::Stop();
   269 
   270 	return KErrNone;
   271 	}
   272 
   273 /**
   274 @SYMTestCaseID SYSLIB-CENTRALREPOSITORY-CT-1419
   275 @SYMTestCaseDesc Test Coarse-Grained Caching functionality.
   276 @SYMTestPriority High
   277 @SYMTestActions  Check correct initialization of the cache manager, expected caching behaviour after
   278  repository open/close, correct setting of cache size, correct functioning of the cache flush
   279  functionality, OOM recovery by cache flushing feature and cache Disable/Enable functionality
   280 @SYMTestExpectedResults The test must not fail.
   281 @SYMPREQ PREQ1192
   282 @SYMTestStatus Defined
   283 @SYMDevelopedForRelease Symbian OS v9.3
   284 @SYMAuthor Aleks Pamir
   285 */
   286 void TRepositoryCacheManagerTester::FuncTestsL()
   287 	{
   288 	NextTest(_L( " @SYMTestCaseID:SYSLIB-CENTRALREPOSITORY-CT-1419 Initialize Cache Manager " ));
   289 
   290 	iTestStepStage = 0;
   291 	TEST( TServerResources::iCacheManager != 0 );
   292 	TEST( TServerResources::iCacheManager->iIdleRepositories.Count() == 0 );
   293 	TServerResources::iCacheManager->EnableCache();
   294 
   295 	NextTest( _L( "Cache Repository (Open/Close)" ) );
   296 
   297 	CacheRepositoryL(KUidCacheTestRepositorySm);
   298 
   299 	if ((iStage == ESizeMin)||(iStage == EReallyWorstCase))
   300 		{
   301 		TEST( TServerResources::iCacheManager->iIdleRepositories.Count() == 0 );
   302 		}
   303 	else
   304 		{
   305 		TEST( TServerResources::iCacheManager->iIdleRepositories.Count() == 1 );
   306 		}
   307 
   308 	NextTest( _L( "Check Cache Size" ) );
   309 
   310 	if ((iStage == ESizeMin)||(iStage == EReallyWorstCase))
   311 		{
   312 		TEST( TServerResources::iCacheManager->iTotalCacheUsage == 0 );
   313 		}
   314 	else
   315 		{
   316 		TEST( TServerResources::iCacheManager->iTotalCacheUsage > 0 );
   317 		}
   318 
   319 	NextTest( _L( "Flush Cache" ) );
   320 
   321 	TServerResources::iCacheManager->FlushCache();
   322 
   323 	TEST( TServerResources::iCacheManager->iIdleRepositories.Count() == 0 );
   324 	TEST( TServerResources::iCacheManager->iTotalCacheUsage == 0 );
   325 
   326 
   327 	if (!iOOMTest)
   328 		{
   329 		// This test is excluded from generic OOM testing because
   330 		// 1. It is trying to simulate OOM condition in a different way (fillingup the memory with
   331 		// allocs) which makes it very tricky to work under normal OOM testing method
   332 		// 2. Loading Lrg repository within OOM takes forever because there're many allocations, and
   333 		// loading other repositories are already tested in other tests
   334 
   335 		NextTest( _L( "Cache OOM Handling" ) );
   336 
   337 		// Fill cache with reps
   338 		TInt totalsize;
   339 		TInt cachedRepositoryCount = 0;
   340 
   341 		totalsize = CacheRepositoryL(KUidCacheTestRepositorySm);
   342 		if (TServerResources::iCacheManager->iCacheSize >= totalsize)
   343 			{
   344 			cachedRepositoryCount++;
   345 			}
   346 		totalsize += CacheRepositoryL(KUidCacheTestRepositorySm2);
   347 		if (TServerResources::iCacheManager->iCacheSize >= totalsize)
   348 			{
   349 			cachedRepositoryCount++;
   350 			}
   351 		totalsize += CacheRepositoryL(KUidCacheTestRepositoryLrg);
   352 		if (TServerResources::iCacheManager->iCacheSize >= totalsize)
   353 			{
   354 			cachedRepositoryCount++;
   355 			}
   356 		totalsize += CacheRepositoryL(KUidCacheTestRepositoryMed2);
   357 		if (TServerResources::iCacheManager->iCacheSize >= totalsize)
   358 			{
   359 			cachedRepositoryCount++;
   360 			}
   361 
   362 		TInt res = KErrNone;
   363 		//	Fill Memory
   364 		TInt popCount=0;
   365 		RHeap& myHeap = User::Heap();
   366 		TInt firstSize = myHeap.Size();
   367 		TInt biggestBlock;
   368 		TInt firstAvail = myHeap.Available(biggestBlock);
   369 
   370 		while(ETrue)
   371 			{
   372 			// We need to really fill up the memory, because we want it to be really freed when we explictly
   373 			// free it, so that when we alloc again there will be some free memory. Using debug allocfail
   374 			// tools would have been too artifical for such a test
   375 			TAny* dummy = User::Alloc(KMemoryFiller);
   376 			if (dummy)
   377 				{
   378 				popCount++;
   379 				TRAP( res, CleanupStack::PushL(dummy); CleanupStack::Pop(dummy););
   380 				if (res == KErrNoMemory)
   381 					{
   382 					// If we cannot allocate enough memory for the cleanupstack frame, we also deallocate
   383 					// the last memory block. This is mandatory for the correct functioning of the
   384 					// following test cases
   385 					// Note when an object that is attempted to be PushL to the cleanupstack and it fails
   386  					// the object will be freed automatically.
   387 					popCount--;
   388 					break;
   389 					}
   390 				else
   391 					{
   392 					CleanupStack::PushL(dummy);
   393 					}
   394 				}
   395 			else
   396 				{
   397 				firstSize = myHeap.Size();
   398 				firstAvail = myHeap.Available(biggestBlock);
   399 				break;
   400 				}
   401 			};
   402 
   403 		// Test if cache is as full as it should be
   404 		TEST( TServerResources::iCacheManager->iIdleRepositories.Count() == cachedRepositoryCount );
   405 
   406 		// Try loading Med rep. Should Fail
   407 		TRAP( res, CacheRepositoryL(KUidCacheTestRepositoryMed));
   408 
   409 		TEST( res == KErrNoMemory );
   410 
   411 		// Cache is still as it was before attempting load
   412 		TEST( TServerResources::iCacheManager->iIdleRepositories.Count() == cachedRepositoryCount );
   413 
   414 		// Try loading Med rep with OOM trapping.
   415 		TBool oomtrap = ETrue;
   416 		TRAP( res, CacheRepositoryL(KUidCacheTestRepositoryMed, oomtrap));
   417 
   418 		if ((iStage == ESizeMin)||(iStage == EReallyWorstCase))
   419 			{
   420 			// Should still fail because the cache was empty to begin with so there's nothing
   421 			// to evict and no memory to gain
   422 			TEST( res == KErrNoMemory );
   423 			TEST( TServerResources::iCacheManager->iIdleRepositories.Count() == 0 );
   424 			}
   425 		else if (iStage == ESizeMinplus)
   426 			{
   427 			// Should still fail because although the cache was full, the amount of memory freed
   428 			// will not be enough for the medium repository
   429 			TEST( res == KErrNoMemory );
   430 			if (oomtrap)
   431 				{
   432 				// Memory alloc failed even before open was attempted, so no cache flush happened
   433 				TEST( TServerResources::iCacheManager->iIdleRepositories.Count() == cachedRepositoryCount );
   434 				}
   435 			else
   436 				{
   437 				TEST( TServerResources::iCacheManager->iIdleRepositories.Count() == 0 );
   438 				}
   439 			}
   440 		else
   441 			{
   442 			// Should Pass and evict repositories
   443 			TEST( res == KErrNone );
   444 			TEST( TServerResources::iCacheManager->iIdleRepositories.Count() == 1 );
   445 			}
   446 
   447 		CleanupStack::PopAndDestroy(popCount);
   448 		}
   449 
   450 	NextTest( _L( "Cache Disable/Enable" ) );
   451 
   452 	TServerResources::iCacheManager->FlushCache();
   453 	TEST( TServerResources::iCacheManager->iIdleRepositories.Count() == 0 );
   454 
   455 	TServerResources::iCacheManager->EnableCache();
   456 
   457 	CacheRepositoryL(KUidCacheTestRepositorySm);
   458 	if ((iStage == ESizeMin)||(iStage == EReallyWorstCase))
   459 		{
   460 		// No cache
   461 		TEST( TServerResources::iCacheManager->iIdleRepositories.Count() == 0 );
   462 		}
   463 	else
   464 		{
   465 		TEST( TServerResources::iCacheManager->iIdleRepositories.Count() == 1 );
   466 		}
   467 
   468 	TServerResources::iCacheManager->DisableCache();
   469 	TEST( TServerResources::iCacheManager->iIdleRepositories.Count() == 0 );
   470 	CacheRepositoryL(KUidCacheTestRepositoryMed);
   471 	TEST( TServerResources::iCacheManager->iIdleRepositories.Count() == 0 );
   472 	TServerResources::iCacheManager->EnableCache();
   473 
   474 	NextTest( _L( "Multi Client Test" ) );
   475 
   476 	TServerResources::iCacheManager->FlushCache();
   477 	TEST( TServerResources::iCacheManager->iIdleRepositories.Count() == 0 );
   478 
   479 	// Creating two sets of server objects
   480 	CSessionNotifier* notifier = new(ELeave)CSessionNotifier;
   481 	CleanupStack::PushL(notifier);
   482 
   483     CServerRepository* repository = new(ELeave) CServerRepository();
   484     CleanupStack::PushL(repository);
   485 
   486 	CSessionNotifier* notifier2 = new(ELeave)CSessionNotifier;
   487 	CleanupStack::PushL(notifier2);
   488 
   489     CServerRepository* repository2 = new(ELeave) CServerRepository();
   490     CleanupStack::PushL(repository2);
   491 
   492     // Open a rep with 1st client
   493     repository->OpenL(KUidCacheTestRepositorySm,*notifier);
   494     // Open same rep with 2nd client
   495     repository2->OpenL(KUidCacheTestRepositorySm,*notifier2);
   496 
   497     // One repository is in the cache, because now(PREQ 1228) all repositories
   498     // are moved to cache just after they are opened, not after they are closed.
   499 	if ((iStage == ESizeMin)||(iStage == EReallyWorstCase))
   500 		{
   501 		// No cache
   502 		TEST( TServerResources::iCacheManager->iIdleRepositories.Count() == 0 );
   503 		}
   504 	else
   505 		{
   506 		TEST( TServerResources::iCacheManager->iIdleRepositories.Count() == 1 );
   507 		}
   508 	// First client closed
   509 	repository->Close();
   510 	// Repository still in the cache
   511 	if ((iStage == ESizeMin)||(iStage == EReallyWorstCase))
   512 		{
   513 		// No cache
   514 		TEST( TServerResources::iCacheManager->iIdleRepositories.Count() == 0 );
   515 		}
   516 	else
   517 		{
   518 		TEST( TServerResources::iCacheManager->iIdleRepositories.Count() == 1 );
   519 		}
   520 	// Second client closed
   521 	repository2->Close();
   522 
   523 	// One rep must still be in the cache now
   524 	if ((iStage == ESizeMin)||(iStage == EReallyWorstCase))
   525 		{
   526 		// No cache
   527 		TEST( TServerResources::iCacheManager->iIdleRepositories.Count() == 0 );
   528 		}
   529 	else
   530 		{
   531 		TEST( TServerResources::iCacheManager->iIdleRepositories.Count() == 1 );
   532 		}
   533 
   534 	NextTest( _L( "Notify-Only Client Cache Repository (Open Only)" ) );
   535 
   536 	TServerResources::iCacheManager->FlushCache();
   537 	TServerResources::iObserver->CloseiOpenRepositories();
   538 
   539     // Open a rep with 1st client
   540     repository->OpenL(KUidCacheTestRepositorySm,*notifier);
   541 
   542 	if ((iStage == ESizeMin)||(iStage == EReallyWorstCase))
   543 		{
   544 		// No memory, so no cache
   545 		TEST( TServerResources::iCacheManager->iIdleRepositories.Count() == 0 );
   546 		// But it's open
   547 		TEST( TServerResources::iObserver->FindOpenRepository(KUidCacheTestRepositorySm) != KErrNotFound );
   548 		}
   549 	else
   550 		{
   551 		// Repository should have been cached after open
   552 		TEST( TServerResources::iCacheManager->iIdleRepositories.Count() == 1 );
   553 		// And it's open
   554 		TEST( TServerResources::iObserver->FindOpenRepository(KUidCacheTestRepositorySm) != KErrNotFound );
   555 		}
   556 
   557 	// First client closed
   558 	repository->Close();
   559 
   560 	if ((iStage == ESizeMin)||(iStage == EReallyWorstCase))
   561 		{
   562 		// No memory, so no cache
   563 		TEST( TServerResources::iCacheManager->iIdleRepositories.Count() == 0 );
   564 		// Now it's closed
   565 		TEST( TServerResources::iObserver->FindOpenRepository(KUidCacheTestRepositorySm) == KErrNotFound );
   566 		}
   567 	else
   568 		{
   569 		// Still in cache because timeout hasn't occured
   570 		TEST( TServerResources::iCacheManager->iIdleRepositories.Count() == 1 );
   571 		// And still open
   572 		TEST( TServerResources::iObserver->FindOpenRepository(KUidCacheTestRepositorySm) != KErrNotFound );
   573 		}
   574 
   575    	CleanupStack::PopAndDestroy(repository2);
   576    	CleanupStack::PopAndDestroy(notifier2);
   577    	CleanupStack::PopAndDestroy(repository);
   578    	CleanupStack::PopAndDestroy(notifier);
   579 
   580 	TServerResources::iCacheManager->FlushCache();
   581 	TEST( TServerResources::iCacheManager->iIdleRepositories.Count() == 0 );
   582 
   583 	}
   584 
   585 /**
   586 @SYMTestCaseID SYSLIB-CENTRALREPOSITORY-CT-1420
   587 @SYMTestCaseDesc Test Coarse-Grained Caching forced eviction rules.
   588 @SYMTestPriority High
   589 @SYMTestActions  Check correct initialization of the cache manager, disabling/enabling
   590  cache with different cache size, filling, eventual forced eviction of repository under
   591  small and large memory cases
   592 @SYMTestExpectedResults The test must not fail.
   593 @SYMPREQ PREQ1192
   594 @SYMTestStatus Defined
   595 @SYMDevelopedForRelease Symbian OS v9.3
   596 @SYMAuthor Aleks Pamir
   597 */
   598 void TRepositoryCacheManagerTester::SizeTestsL()
   599 	{
   600 	NextTest( _L( " @SYMTestCaseID:SYSLIB-CENTRALREPOSITORY-CT-1420 Forced Eviction-Small cache " ) );
   601 
   602 	iTestStepStage = 1;
   603 	TEST( TServerResources::iCacheManager != 0 );
   604 	TEST( TServerResources::iCacheManager->iIdleRepositories.Count() == 0 );
   605 
   606 	// Flush and disable the cache
   607 	TServerResources::iCacheManager->DisableCache(ETrue);
   608 
   609 	// Calculate cache size for this test. It's important to calculate this at runtime,
   610 	// because this size is approximate and it changes from emulator to hardware
   611 
   612 	TServerResources::iCacheManager->EnableCache(KInvalidEvictionTimeout, KDefaultCacheSize);
   613 
   614 	TInt smallCacheSize = CacheRepositoryL(KUidCacheTestRepositorySm);
   615 	smallCacheSize+= CacheRepositoryL(KUidCacheTestRepositoryMed);
   616 	smallCacheSize+= (CacheRepositoryL(KUidCacheTestRepositorySm2)/2);
   617 
   618 	// smallCacheSize is KUidCacheTestRepositorySm+KUidCacheTestRepositoryMed+ half of KUidCacheTestRepositorySm2
   619 	// the reason is we don't want KUidCacheTestRepositorySm2 to fit
   620 
   621 	TServerResources::iCacheManager->DisableCache(ETrue);
   622 	//Change cache size
   623 	TServerResources::iCacheManager->EnableCache(KInvalidEvictionTimeout, smallCacheSize);
   624 
   625 	// Fill cache
   626 	CacheRepositoryL(KUidCacheTestRepositorySm);
   627 	CacheRepositoryL(KUidCacheTestRepositoryMed);
   628 	User::After(KTimerDelay);
   629 	//This one will not fit, and the Small rep will be evicted because it's small
   630 	CacheRepositoryL(KUidCacheTestRepositorySm2);
   631 
   632 	TEST( !FindRepository(KUidCacheTestRepositorySm) );
   633 	TEST( FindRepository(KUidCacheTestRepositorySm2) );
   634 
   635 	TInt delay = TServerResources::iCacheManager->iDefaultTimeout.Int()-KTimerDelay;
   636 	// If timeout is smaller then KTimerDelay, don't bother to wait
   637 	if (delay>0)
   638 		{
   639 		User::After(delay);
   640 		}
   641 	//Medium will be evicted this time, because it's older and size difference doesn't make up for the time difference
   642 	CacheRepositoryL(KUidCacheTestRepositorySm);
   643 
   644 	TEST( !FindRepository(KUidCacheTestRepositoryMed) );
   645 	TEST( FindRepository(KUidCacheTestRepositorySm) );
   646 
   647 	if (!iOOMTest)
   648 		{
   649 		// This test is excluded from generic OOM testing because
   650 		// 1. Loading Lrg repository within OOM takes forever because there're many allocations, and
   651 		// the test is very similar to previous test
   652 
   653 		NextTest( _L( "Forced Eviction-Large cache" ) );
   654 
   655 		TServerResources::iCacheManager->DisableCache(ETrue);
   656 
   657 		TServerResources::iCacheManager->EnableCache(KInvalidEvictionTimeout, KDefaultCacheSize);
   658 
   659 		TInt largeCacheSize = CacheRepositoryL(KUidCacheTestRepositoryMed);
   660 		largeCacheSize+= CacheRepositoryL(KUidCacheTestRepositoryLrg);
   661 		largeCacheSize+= (CacheRepositoryL(KUidCacheTestRepositoryMed2)/2);
   662 
   663 		// smallCacheSize is KUidCacheTestRepositorySm+KUidCacheTestRepositoryMed+ half of KUidCacheTestRepositorySm2
   664 		// the reason is we don't want KUidCacheTestRepositorySm2 to fit
   665 
   666 		TServerResources::iCacheManager->DisableCache(ETrue);
   667 
   668 		//Change Parameters
   669 		TServerResources::iCacheManager->EnableCache(KInvalidEvictionTimeout, largeCacheSize);
   670 		// Fill cache
   671 		CacheRepositoryL(KUidCacheTestRepositoryMed);
   672 		CacheRepositoryL(KUidCacheTestRepositoryLrg);
   673 		User::After(KTimerDelay);
   674 		//This one will not fit, and the Med rep will be evicted because it's small
   675 		CacheRepositoryL(KUidCacheTestRepositoryMed2);
   676 
   677 		TEST( !FindRepository(KUidCacheTestRepositoryMed) );
   678 		TEST( FindRepository(KUidCacheTestRepositoryLrg) );
   679 
   680 		delay = TServerResources::iCacheManager->iDefaultTimeout.Int()-KTimerDelay;
   681 		// If timeout is smaller then KTimerDelay, don't bother to wait
   682 		if (delay>0)
   683 			{
   684 			User::After(delay);
   685 			}
   686 
   687 		//Medium2 will be evicted, because even if it's new, this time size difference makes up for the
   688 		//time difference and the Large repository stays even if it's older
   689 		CacheRepositoryL(KUidCacheTestRepositoryMed);
   690 
   691 		TEST( !FindRepository(KUidCacheTestRepositoryMed2) );
   692 		TEST( FindRepository(KUidCacheTestRepositoryLrg) );
   693 		}
   694 	}
   695 
   696 /**
   697 @SYMTestCaseID SYSLIB-CENTRALREPOSITORY-CT-1421
   698 @SYMTestCaseDesc Test Coarse-Grained Caching normal eviction rules.
   699 @SYMTestPriority High
   700 @SYMTestActions  Check correct initialization of the cache manager, adding repository into
   701  cache and active object driven eviction at correct times.
   702 @SYMTestExpectedResults The test must not fail.
   703 @SYMPREQ PREQ1192
   704 @SYMTestStatus Defined
   705 @SYMDevelopedForRelease Symbian OS v9.3
   706 @SYMAuthor Aleks Pamir
   707 */
   708 void TRepositoryCacheManagerTester::TimingTestsL()
   709 	{
   710 
   711 	iTestStepStage = 2;
   712 
   713 	TEST( TServerResources::iCacheManager != 0 );
   714 	TServerResources::iCacheManager->FlushCache();
   715 	TEST( TServerResources::iCacheManager->iIdleRepositories.Count() == 0 );
   716 
   717 	CPeriodic* timer = CPeriodic::NewL(EPriorityLow);
   718 	CleanupStack::PushL(timer);
   719 
   720 	NextTest( _L( " @SYMTestCaseID:SYSLIB-CENTRALREPOSITORY-CT-1421 Cache Timing(Component Test) " ) );
   721 
   722 	starttime.UniversalTime();
   723 	// Add one repository in the cache
   724 	CacheRepositoryL(KUidCacheTestRepositorySm);
   725 	TTime now;
   726 	now.UniversalTime();
   727 	TheTest.Printf(_L("KUidCacheTestRepositorySm Added at %Ld\n"), now.MicroSecondsFrom(starttime).Int64());
   728 
   729 	TEST( TServerResources::iCacheManager->iIdleRepositories.Count() == 1 );
   730 
   731 	// Wait until the repository is evicted normally
   732 	// We can't use User::After because we don't want to suspend the thread
   733 	timer->Start(TTimeIntervalMicroSeconds32(TServerResources::iCacheManager->iDefaultTimeout.Int()-KTimerDelay), 0, TCallBack(Callback, timer));
   734 
   735 	CActiveScheduler::Start();
   736 
   737 	// Callback should have been called before Cache timeout had occured
   738 	TEST( TServerResources::iCacheManager->iIdleRepositories.Count() == 1 );
   739 
   740 	TServerResources::iCacheManager->FlushCache();
   741 	TEST( TServerResources::iCacheManager->iIdleRepositories.Count() == 0 );
   742 
   743 	starttime.UniversalTime();
   744 	// Add another repository in the cache
   745 	CacheRepositoryL(KUidCacheTestRepositorySm2);
   746 	now.UniversalTime();
   747 	TheTest.Printf(_L("KUidCacheTestRepositorySm2 added at %Ld\n"), now.MicroSecondsFrom(starttime).Int64());
   748 
   749 	// wait more than default timeout.
   750 	timer->Start(TTimeIntervalMicroSeconds32(TServerResources::iCacheManager->iDefaultTimeout.Int()+KTimerDelay), 0, TCallBack(Callback, timer));
   751 
   752 	CActiveScheduler::Start();
   753 
   754 	// Callback should have been called by now
   755 	TInt count = TServerResources::iCacheManager->iIdleRepositories.Count();
   756 	if (count > 0)
   757 		{
   758 		// Do not fail test on emulator. CTimer::AtUTC is often
   759 		// late by 1 to a few seconds.
   760 		#if defined(__WINSCW__) || defined(__WINS__)
   761 		TheTest.Printf(_L("*** Line %d check fail. Ignored on winscw."), __LINE__);
   762 		TServerResources::iCacheManager->FlushCache();
   763 		#endif
   764 		}
   765 
   766 	TEST( TServerResources::iCacheManager->iIdleRepositories.Count() == 0 );
   767 
   768 	TServerResources::iCacheManager->FlushCache();
   769 	TEST( TServerResources::iCacheManager->iIdleRepositories.Count() == 0 );
   770 
   771 	NextTest( _L( "Notify-Only Client Timing Test" ) );
   772 
   773 	// Add one repository in the cache
   774 	CSessionNotifier* notifier = new(ELeave)CSessionNotifier;
   775 	CleanupStack::PushL(notifier);
   776 
   777     CServerRepository* repository = new(ELeave) CServerRepository();
   778     CleanupStack::PushL(repository);
   779 
   780 	starttime.UniversalTime();
   781 
   782     // Open a rep
   783     repository->OpenL(KUidCacheTestRepositorySm,*notifier);
   784 
   785 	now.UniversalTime();
   786 	TheTest.Printf(_L("KUidCacheTestRepositorySm Added at %Ld\n"), now.MicroSecondsFrom(starttime).Int64());
   787 
   788 	TEST( TServerResources::iCacheManager->iIdleRepositories.Count() == 1 );
   789 
   790 	// Wait until the repository is evicted normally
   791 	// We can't use User::After because we don't want to suspend the thread
   792 	timer->Start(TTimeIntervalMicroSeconds32(TServerResources::iCacheManager->iDefaultTimeout.Int()-KTimerDelay), 0, TCallBack(Callback, timer));
   793 
   794 	CActiveScheduler::Start();
   795 
   796 	// Callback should have been called before Cache timeout had occured
   797 	TEST( TServerResources::iCacheManager->iIdleRepositories.Count() == 1 );
   798 
   799 	TServerResources::iCacheManager->FlushCache();
   800 	TEST( TServerResources::iCacheManager->iIdleRepositories.Count() == 0 );
   801 
   802 	// Add another repository in the cache
   803 	// Add one repository in the cache
   804 	CSessionNotifier* notifier2 = new(ELeave)CSessionNotifier;
   805 	CleanupStack::PushL(notifier2);
   806 
   807     CServerRepository* repository2 = new(ELeave) CServerRepository();
   808     CleanupStack::PushL(repository2);
   809 
   810 	starttime.UniversalTime();
   811 
   812     // Open another rep
   813     repository2->OpenL(KUidCacheTestRepositorySm2,*notifier2);
   814 
   815 	now.UniversalTime();
   816 	TheTest.Printf(_L("KUidCacheTestRepositorySm2 added at %Ld\n"), now.MicroSecondsFrom(starttime).Int64());
   817 
   818 	// wait more than default timeout.
   819 	timer->Start(TTimeIntervalMicroSeconds32(TServerResources::iCacheManager->iDefaultTimeout.Int()+KTimerDelay), 0, TCallBack(Callback, timer));
   820 
   821 	CActiveScheduler::Start();
   822 
   823 	// Callback should have been called by now
   824 	count = TServerResources::iCacheManager->iIdleRepositories.Count();
   825 	if (count > 0)
   826 		{
   827 		// Do not fail test on emulator. CTimer::AtUTC is often
   828 		// late by 1 to a few seconds.
   829 		#if defined(__WINSCW__) || defined(__WINS__)
   830 		TheTest.Printf(_L("*** Line %d check fail. Ignored on winscw."), __LINE__);
   831 		TServerResources::iCacheManager->FlushCache();
   832 		#endif
   833 		}
   834 
   835 	TEST( TServerResources::iCacheManager->iIdleRepositories.Count() == 0 );
   836 
   837 	repository2->Close();
   838 	repository->Close();
   839 
   840    	CleanupStack::PopAndDestroy(repository2);
   841    	CleanupStack::PopAndDestroy(notifier2);
   842    	CleanupStack::PopAndDestroy(repository);
   843    	CleanupStack::PopAndDestroy(notifier);
   844 
   845 	CleanupStack::PopAndDestroy(timer);
   846 	}
   847 
   848 void TRepositoryCacheManagerTester::DefectTestsL()
   849 	{
   850 	iTestStepStage = 3;	// Don't run OOM tests on defects
   851 
   852 	NextTest(_L( "DEF093491: [AQP]Centrep server flushes repository cache when temporarily disabled" ));
   853    	DEF093491L();
   854 
   855 	NextTest(_L( "INC105967: CenRep crashes when no ROM directory" ));
   856    	INC105967();
   857    	
   858 	}
   859 
   860 /**
   861 @SYMTestCaseID SYSLIB-CENTRALREPOSITORY-CT-1883
   862 @SYMTestCaseDesc [AQP]Centrep server flushes repository cache when temporarily disabled
   863 @SYMTestPriority High
   864 @SYMTestActions  Open repository, disable cache, check if the rep is still in memory
   865 @SYMTestExpectedResults The test must not fail or panic .
   866 @SYMDEF DEF093491
   867 */
   868 void TRepositoryCacheManagerTester::DEF093491L()
   869 	{
   870 	NextTest( _L( " @SYMTestCaseID:SYSLIB-CENTRALREPOSITORY-CT-1883 " ) );
   871 	TServerResources::iCacheManager->FlushCache();
   872 
   873 	// Notifier needed to open repositories.
   874 	CSessionNotifier* notifier;
   875 	notifier = new(ELeave)CSessionNotifier;
   876 	CleanupStack::PushL(notifier);
   877 
   878     CServerRepository* repository = new(ELeave) CServerRepository();
   879     CleanupStack::PushL(repository);
   880     // Open repository
   881     repository->OpenL(KUidCacheTestRepositorySm,*notifier);
   882 
   883 	// Check it's in cache and memory
   884 	TEST( TServerResources::iCacheManager->iIdleRepositories.Count() == 1 );
   885 	TEST( TServerResources::iObserver->FindOpenRepository(KUidCacheTestRepositorySm) != KErrNotFound );
   886     // disable cache
   887 	TServerResources::iCacheManager->DisableCache();
   888 
   889 	// Check it's in memory but not in cache
   890 	TEST( TServerResources::iCacheManager->iIdleRepositories.Count() == 0 );
   891 	TEST( TServerResources::iObserver->FindOpenRepository(KUidCacheTestRepositorySm) != KErrNotFound );
   892 
   893 	repository->Close();
   894 	// the repository should now be unloaded from memory
   895 	TEST( TServerResources::iObserver->FindOpenRepository(KUidCacheTestRepositorySm) == KErrNotFound );
   896 	// enable cache again
   897 	TServerResources::iCacheManager->EnableCache();
   898 
   899     // Open repository again
   900     repository->OpenL(KUidCacheTestRepositorySm,*notifier);
   901 
   902 	// Check it's in cache and memory
   903 	TEST( TServerResources::iCacheManager->iIdleRepositories.Count() == 1 );
   904 	TEST( TServerResources::iObserver->FindOpenRepository(KUidCacheTestRepositorySm) != KErrNotFound );
   905 
   906 	repository->Close();
   907 
   908     // disable cache
   909 	TServerResources::iCacheManager->DisableCache(ETrue);
   910 
   911 	// now it should be flushed out of both memory and cache
   912 	TEST( TServerResources::iCacheManager->iIdleRepositories.Count() == 0 );
   913 	TEST( TServerResources::iObserver->FindOpenRepository(KUidCacheTestRepositorySm) == KErrNotFound );
   914 
   915    	CleanupStack::PopAndDestroy(repository);
   916    	CleanupStack::PopAndDestroy(notifier);
   917 	}
   918 
   919 // Helper function for INC105967L
   920 LOCAL_C TInt TestThread(TAny*)
   921 	{
   922 	CTrapCleanup* cleanup = CTrapCleanup::New();
   923 	if(!cleanup)
   924 		return KErrNoMemory;
   925 
   926 	HBufC* tempFileName;
   927 	// this call shouldn't cause a panic, but leave with KErrNotFound
   928 	TRAPD(err,
   929 		{
   930 		TServerResources::CreateRepositoryFileNameLC(tempFileName, KUidCacheTestRepositorySm, ERom, ECre);
   931 		CleanupStack::PopAndDestroy(tempFileName);
   932 		});
   933 	// test if the function leaves with KErrNot Found
   934 	TEST2(err, KErrNotFound);
   935 
   936 	delete cleanup;
   937 	return KErrNone;
   938 	}
   939 
   940 /**
   941 @SYMTestCaseID SYSLIB-CENTRALREPOSITORY-UT-3474
   942 @SYMTestCaseDesc CenRep crashes when no ROM directory
   943 @SYMTestPriority High
   944 @SYMTestActions Save RomDirectory, temporarily NULL the pointer, call panicking function in a seperate thread
   945  (to survive even if it panics and continue other tests), check that it leaves with KErrNotFound, restore RomDirectory
   946 @SYMTestExpectedResults The test must not panic.
   947 @SYMDEF INC105967
   948 */
   949 void TRepositoryCacheManagerTester::INC105967()
   950 	{
   951 	NextTest( _L( " @SYMTestCaseID:SYSLIB-CENTRALREPOSITORY-UT-3474 " ) );
   952 	//store RomDirectory
   953 	HBufC* tempRomDirectory = TServerResources::iRomDirectory;
   954 
   955 	// temporarily delete it to simulate the behaviour in TServerResources::InitialiseL() when:
   956 	// 	if(iFs.Entry(*iRomDirectory, fsEntry)!=KErrNone || !fsEntry.IsDir())
   957 	TServerResources::iRomDirectory = NULL;
   958 
   959 	RThread testThread;
   960 	_LIT(KThreadName, "TestThread");
   961 
   962 	testThread.Create(KThreadName, TestThread, KDefaultStackSize, KMinHeapSize, 0x100000, NULL);
   963 
   964 	TRequestStatus requestStatus;
   965 	// Request notification when the thread terminates
   966 	testThread.Logon(requestStatus);
   967 	// Let the thread execute
   968 	testThread.Resume();
   969 
   970 	// Wait for termination
   971 	User::WaitForRequest(requestStatus);
   972 	// test if the thread terminated normally (the leave has been trapped) not panicked
   973 	TEST2(requestStatus.Int(), KErrNone);
   974 
   975 	//restore RomDirectory
   976 	TServerResources::iRomDirectory = tempRomDirectory;
   977 	}
   978 
   979 /**
   980 @SYMTestCaseID PDS-CENTRALREPOSITORY-UT-4077
   981 @SYMTestCaseDesc Central Repository cache manager incorrectly uses CTimer::AtUTC() 
   982 @SYMTestPriority High
   983 @SYMTestActions Call CRepositoryCacheManager::RescheduleTimer passing in various
   984 				timer values to test conversion of 64 bit UTC time value into
   985 				32 bit microsecond value.
   986 @SYMTestExpectedResults The timer should be active after each call to RescheduleTimer
   987 						The test must not panic.
   988 @SYMDEF DEF124147
   989 */
   990 void TRepositoryCacheManagerTester::DEF124147()
   991 	{
   992 	NextTest( _L( " @SYMTestCaseID:PDS-CENTRALREPOSITORY-UT-4077 " ) );
   993 	
   994 	//Cancel any pending timer
   995 	TServerResources::iCacheManager->Cancel();
   996 	
   997 	TTime now;
   998 	now.UniversalTime();
   999 	
  1000 	TTimeIntervalMinutes oneMinute(1);
  1001 	TTimeIntervalHours oneHour(1);
  1002 	
  1003 	//Rechedule timer now
  1004 	TServerResources::iCacheManager->RescheduleTimer(now);
  1005 	TEST(TServerResources::iCacheManager->IsActive());
  1006 	
  1007 	//Cancel any pending timer
  1008 	TServerResources::iCacheManager->Cancel();
  1009 	
  1010 	
  1011 	//Rechedule timer in the past
  1012 	TServerResources::iCacheManager->RescheduleTimer(now - oneMinute);
  1013 	TEST(TServerResources::iCacheManager->IsActive());
  1014 	
  1015 	//Cancel any pending timer
  1016 	TServerResources::iCacheManager->Cancel();
  1017 	
  1018 	//Rechedule timer in the future
  1019 	TServerResources::iCacheManager->RescheduleTimer(now + oneMinute);
  1020 	TEST(TServerResources::iCacheManager->IsActive());
  1021 	
  1022 	//Cancel any pending timer
  1023 	TServerResources::iCacheManager->Cancel();
  1024 	
  1025 	//Rechedule timer an hour in the past
  1026 	TServerResources::iCacheManager->RescheduleTimer(now - oneHour);
  1027 	TEST(TServerResources::iCacheManager->IsActive());
  1028 	
  1029 	//Cancel any pending timer
  1030 	TServerResources::iCacheManager->Cancel();
  1031 	
  1032 	//Rechedule timer an hour in the future
  1033 	TServerResources::iCacheManager->RescheduleTimer(now + oneHour);
  1034 	TEST(TServerResources::iCacheManager->IsActive());
  1035 	
  1036 	//Cancel any pending timer
  1037 	TServerResources::iCacheManager->Cancel();
  1038 	}
  1039 
  1040 // ---------------------------------------------------
  1041 // OomTest
  1042 //
  1043 // Function to convert a test into an OOM test
  1044 
  1045 /**
  1046 @SYMTestCaseID SYSLIB-CENTRALREPOSITORY-CT-1422
  1047 @SYMTestCaseDesc Test functionality under OOM.
  1048 @SYMTestPriority High
  1049 @SYMTestActions  Test functionality under OOM.
  1050 @SYMTestExpectedResults The test must not fail.
  1051 @SYMPREQ PREQ1192
  1052 @SYMTestStatus Defined
  1053 @SYMDevelopedForRelease Symbian OS v9.3
  1054 @SYMAuthor Aleks Pamir
  1055 */
  1056 LOCAL_C void OomTest( void (*testFuncL)() )
  1057 	{
  1058 	TheTest.Next( _L( " @SYMTestCaseID:SYSLIB-CENTRALREPOSITORY-CT-1422 " ) );
  1059 	TInt error;
  1060 	TInt count = 0;
  1061 
  1062 	do
  1063 		{
  1064 		User::__DbgMarkStart( RHeap::EUser );
  1065 
  1066 		// find out the number of open handles
  1067 		TInt startProcessHandleCount;
  1068 		TInt startThreadHandleCount;
  1069 		RThread().HandleCount(startProcessHandleCount, startThreadHandleCount);
  1070 
  1071 		User::__DbgSetAllocFail( RHeap::EUser, RHeap::EFailNext, ++count );
  1072 
  1073 		TRAP( error, (testFuncL)() );
  1074 
  1075 		User::__DbgSetAllocFail( RHeap::EUser, RHeap::ENone, 1 );
  1076 
  1077 		// check that no handles have leaked
  1078 		TInt endProcessHandleCount;
  1079 		TInt endThreadHandleCount;
  1080 		RThread().HandleCount(endProcessHandleCount, endThreadHandleCount);
  1081 
  1082 		TEST(startProcessHandleCount == endProcessHandleCount);
  1083 		TEST(startThreadHandleCount  == endThreadHandleCount);
  1084 
  1085 		User::__DbgMarkEnd( RHeap::EUser, 0 );
  1086 		} while( error == KErrNoMemory );
  1087 
  1088 	_LIT( KTestFailed, "Out of memory test failure on iteration %d\n" );
  1089 	__ASSERT_ALWAYS( error == KErrNone, TheTest.Panic( error, KTestFailed, count ) );
  1090 
  1091 	TheTest.Printf( _L( "Out of memory tests succeeded at heap failure rate of %i\n" ), count );
  1092 	TheTest.Printf( _L( "Process handle count preserved\n" ) );
  1093 	TheTest.Printf( _L( "Thread handle count preserved\n" ) );
  1094 	}
  1095 
  1096 LOCAL_C void DoOOMTestsL()
  1097 	{
  1098 	// To clean up the static array under OOM
  1099 	CleanupStack::PushL(gCleanup);
  1100 
  1101 	// To be able to run tests faster, especially the last 2 tests with delays,
  1102 	// a flow-through to the next case statement is intended here. If the Tester
  1103 	// function completes successfully, meaning that it has passed all memory
  1104 	// allocations, then the execution continues from the next step and the iTestStepStage
  1105 	// variable is set to the current test step stage. Otherwise the function will Leave
  1106 	// with KErrNoMemory, will be trapped in OomTest and DoOOMTestsL will be recalled.
  1107 	// Once a test step stage is passed, Tester.iTestStepStage will ensure that the same
  1108 	// function will no be called over and over again for every memory allocation failure,
  1109 	// thereby reducing the running time of the test considerably.
  1110 
  1111 	switch(Tester.iTestStepStage)
  1112 		{
  1113 		case 0:
  1114 			{
  1115 			Tester.FuncTestsL();
  1116 			}
  1117 		case 1:
  1118 			{
  1119 			Tester.SizeTestsL();
  1120 			}
  1121 		case 2:
  1122 			{
  1123 			Tester.TimingTestsL();
  1124 			}
  1125 		default:
  1126 			break;
  1127 		}
  1128 	// To clean up the memory left in static variables
  1129 	Tester.CleanUp(NULL);
  1130 
  1131 	CleanupStack::Pop();
  1132 	}
  1133 
  1134 LOCAL_C void DoTestsL()
  1135 	{
  1136 	switch(Tester.iStage)
  1137 		{
  1138 		case ENominalNoFile:
  1139 		default:
  1140 			{
  1141 			TheTest.Start( _L( "Cache functionality tests with default params(no .ini)" ) );
  1142 			Tester.FuncTestsL();
  1143 			Tester.SizeTestsL();
  1144 			Tester.TimingTestsL();
  1145 			TheTest.End();
  1146 			break;
  1147 			}
  1148 		case ENominal:
  1149 			{
  1150 			TheTest.Start( _L( "Cache functionality tests with default params(default values read from .ini)" ) );
  1151 			Tester.FuncTestsL();
  1152 			Tester.SizeTestsL();
  1153 			Tester.TimingTestsL();
  1154 			// Only test defects once here unless a defect occurs in different memory conditions
  1155 			Tester.DefectTestsL();
  1156 			TheTest.End();
  1157 			break;
  1158 			}
  1159 		case ESizeMin:
  1160 			{
  1161 			TheTest.Start( _L( "Cache functionality tests with no space for cache (size=0)" ) );
  1162 			Tester.FuncTestsL();
  1163 			// No size tests because size is controlled in this test
  1164 			// No Timing tests because nothing will be cached(cache size 0) so nothing to time
  1165 			TheTest.End();
  1166 			break;
  1167 			}
  1168 		case ESizeMinplus:
  1169 			{
  1170 			TheTest.Start( _L( "Cache functionality tests with small cache size" ) );
  1171 			Tester.FuncTestsL();
  1172 			// No size tests because size is controlled in this test
  1173 			Tester.TimingTestsL();
  1174 			TheTest.End();
  1175 			break;
  1176 			}
  1177 		case ESizeMaxminus:
  1178 			{
  1179 			TheTest.Start( _L( "Cache functionality tests with large cache size" ) );
  1180 			Tester.FuncTestsL();
  1181 			// No size tests because size is controlled in this test
  1182 			Tester.TimingTestsL();
  1183 			TheTest.End();
  1184 			break;
  1185 			}
  1186 		case ESizeMax:
  1187 			{
  1188 			TheTest.Start( _L( "Cache functionality tests with max cache size (=heap max 2MB)" ) );
  1189 			Tester.FuncTestsL();
  1190 			// No size tests because size is controlled in this test
  1191 			Tester.TimingTestsL();
  1192 			TheTest.End();
  1193 			break;
  1194 			}
  1195 		case ESizeMaxplus:
  1196 			{
  1197 			TheTest.Start( _L( "Cache functionality Robustness test" ) );
  1198 			Tester.FuncTestsL();
  1199 			// No size tests because size is controlled in this test
  1200 			Tester.TimingTestsL();
  1201 			TheTest.End();
  1202 			break;
  1203 			}
  1204 		case ETimeoutMin:
  1205 			{
  1206 			TheTest.Start( _L( "Cache functionality tests with no timeout for cache (timeout=0)" ) );
  1207 			Tester.FuncTestsL();
  1208 			Tester.SizeTestsL();
  1209 			// No timing tests because timeout is controlled in this test
  1210 			TheTest.End();
  1211 			break;
  1212 			}
  1213 		case ETimeoutMinplus:
  1214 			{
  1215 			TheTest.Start( _L( "Cache functionality tests with short timeout" ) );
  1216 			Tester.FuncTestsL();
  1217 			Tester.SizeTestsL();
  1218 			// No timing tests because timeout is controlled in this test
  1219 			TheTest.End();
  1220 			break;
  1221 			}
  1222 		case ETimeoutMaxminus:
  1223 			{
  1224 			TheTest.Start( _L( "Cache functionality tests with large timeout" ) );
  1225 			Tester.FuncTestsL();
  1226 			Tester.SizeTestsL();
  1227 			// No timing tests because timeout is controlled in this test
  1228 			TheTest.End();
  1229 			break;
  1230 			}
  1231 		case EReallyWorstCase:
  1232 			{
  1233 			TheTest.Start( _L( "Cache functionality Worst Case test with no timeout and no size for cache" ) );
  1234 			Tester.FuncTestsL();
  1235 			Tester.SizeTestsL();
  1236 			// No Timing tests because nothing will be cached(cache size 0) so nothing to time
  1237 			TheTest.End();
  1238 			break;
  1239 			}
  1240 		};
  1241 	}
  1242 
  1243 
  1244 /**
  1245 @SYMTestCaseID SYSLIB-CENTRALREPOSITORY-UT-4014
  1246 @SYMTestCaseDesc Cenrep cache manager should not evict sub-sessions in active transactions.
  1247 @SYMTestPriority High
  1248 @SYMTestActions  Open 2 sub-sessions to a repository, start a transaction, let the cache manager run.
  1249 @SYMTestExpectedResults The test must not fail or panic .
  1250 @SYMDEF DEF111734
  1251 */
  1252 void TRepositoryCacheManagerTester::DEF111734L()
  1253 	{
  1254 	NextTest( _L( " @SYMTestCaseID:SYSLIB-CENTRALREPOSITORY-UT-4014 DEF111734: Cache Manager and Open Transactions "));
  1255 
  1256 	TServerResources::iCacheManager->FlushCache();
  1257 
  1258 	// Notifier needed to open repositories.
  1259 	// notifier 1
  1260 	CSessionNotifier* notifier;
  1261 	notifier = new(ELeave)CSessionNotifier;
  1262 	CleanupStack::PushL(notifier);
  1263 
  1264 	// notifier 2
  1265 	CSessionNotifier* notifier2;
  1266 	notifier2 = new(ELeave)CSessionNotifier;
  1267 	CleanupStack::PushL(notifier2);
  1268 
  1269 	// connection 1
  1270  	CServerRepository* repository = new(ELeave) CServerRepository();
  1271 	CleanupStack::PushL(repository);
  1272 
  1273 	// connection 2
  1274 	CServerRepository* repository2 = new(ELeave) CServerRepository();
  1275  	CleanupStack::PushL(repository2);
  1276 
  1277 	// Open repository
  1278 	repository->OpenL(KUidCacheTestRepositorySm,*notifier);
  1279 
  1280 	// open second connection to the repository
  1281 	repository2->OpenL(KUidCacheTestRepositorySm,*notifier2);
  1282 
  1283 	// we have to observers to the same repository - so we should have only 1 entry in
  1284 	// the idle repositories for the cache manager
  1285 
  1286 	TEST(TServerResources::iCacheManager->iIdleRepositories.Count() == 1);
  1287 
  1288 	// check CR's global memory for an instance for this repository.
  1289 	TEST(TServerResources::iObserver->FindOpenRepository(KUidCacheTestRepositorySm) != KErrNotFound);
  1290 
  1291 	// start a transaction on connection 2, this adds this sub-session to the
  1292 	// transaction queue for the repository.
  1293 
  1294 	TInt error = repository2->StartTransaction(3);
  1295 	TEST(error == KErrNone);
  1296 
  1297 	// This will wipe out the memory for the open repository unless we modify the code in some fashion.
  1298 	TServerResources::iCacheManager->FlushCache(EFalse);
  1299 
  1300 	// since we can't use the user::after method since we need to this thread to suspend
  1301 	// start a timer (stolen from above).
  1302 	CPeriodic* timer = CPeriodic::NewL(EPriorityLow);
  1303 	CleanupStack::PushL(timer);
  1304 
  1305 
  1306 	// since repositories that involved in active transactions no longer have their memory
  1307 	// removed by cache manager, give this time enough time to go through a few cycles of
  1308 	// the normal eviction process.
  1309 
  1310 	timer->Start(TTimeIntervalMicroSeconds32(TServerResources::iCacheManager->iDefaultTimeout.Int()*3), 0, TCallBack(Callback, timer));
  1311 
  1312 	// will cause the cache manager to run
  1313 	CActiveScheduler::Start();
  1314 
  1315 	// If it isn't in cache and memory than the cache manager must have collected it
  1316 	// and the code for this defect is not in the build.
  1317 	TEST(TServerResources::iCacheManager->iIdleRepositories.Count() == 0);
  1318 	TEST(TServerResources::iObserver->FindOpenRepository(KUidCacheTestRepositorySm) != KErrNotFound);
  1319 
  1320 	// close this sub-session.  should cancel the active transaction. allowing cache manager to clean
  1321 	// up the memory.
  1322 	repository2->Close();
  1323 
  1324 	TInt i = 0;
  1325 	const TUint32 KInt1 = 0x1;
  1326 	repository->Get(KInt1, i);
  1327 
  1328 	// close the last observer for this repository.  this places the memory for this repository
  1329 	// back onto the idle list for cache manager.
  1330 	repository->Close();
  1331 
  1332 	// Wait until the repository is evicted normally
  1333 
  1334 	timer->Start(TTimeIntervalMicroSeconds32(TServerResources::iCacheManager->iDefaultTimeout.Int()*2), 0, TCallBack(Callback, timer));
  1335 
  1336 	CActiveScheduler::Start();
  1337 
  1338 	// the repository should now be unloaded from memory
  1339 	TEST( TServerResources::iObserver->FindOpenRepository(KUidCacheTestRepositorySm) == KErrNotFound );
  1340 
  1341    	CleanupStack::PopAndDestroy(5);
  1342 	}
  1343 
  1344 LOCAL_C void MainL()
  1345 	{
  1346 	__UHEAP_MARK;
  1347 	TheTest.Start( _L( "Cache Tests" ) );
  1348 
  1349 	// create and install the active scheduler we need
  1350 	CActiveScheduler* s = new(ELeave) CActiveScheduler;
  1351 	CleanupStack::PushL( s );
  1352 	CActiveScheduler::Install( s );
  1353 
  1354 	Tester.DeleteFilesL();
  1355 
  1356 	TServerResources::InitialiseL();
  1357 
  1358 	
  1359 	Tester.DEF111734L();
  1360 	Tester.DEF124147();
  1361 	
  1362 
  1363 	for(TInt i=ENominalNoFile; i<ELastStage; i++)
  1364 		{
  1365 		Tester.AdvanceToStageL( static_cast<TBorderTestStage>(i) );
  1366 		DoTestsL();
  1367 		}
  1368 
  1369 	TheTest.Next( _L(  "Out of memory tests" ) );
  1370 	Tester.iOOMTest = ETrue;
  1371 
  1372 	Tester.AdvanceToStageL( ENominal );
  1373 	OomTest( DoOOMTestsL );
  1374 
  1375 	TServerResources::Close();
  1376 
  1377 	// Cleanup the scheduler
  1378 	CleanupStack::PopAndDestroy( s );
  1379 
  1380 	TheTest.End();
  1381 	TheTest.Close();
  1382 	__UHEAP_MARKEND;
  1383 	}
  1384 
  1385 TInt E32Main()
  1386 	{
  1387 #ifdef __SECURE_DATA__
  1388 	__UHEAP_MARK;
  1389 	CTrapCleanup* cleanup = CTrapCleanup::New();
  1390 	if ( !cleanup )
  1391 		return KErrNoMemory;
  1392 
  1393 	TRAPD( err, MainL()  );
  1394 	if ( err != KErrNone )
  1395 		User::Panic( _L( "Testing failed: " ), err );
  1396 
  1397 	delete cleanup;
  1398 	__UHEAP_MARKEND;
  1399 #endif
  1400 
  1401 	return 0;
  1402 	}
  1403