os/ossrv/lowlevellibsandfws/pluginfw/Framework/ResolverTest/t_resolvercache.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) 2008-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
// This file contains test classes and their implementations
sl@0
    15
// to test production class CCustomResolverCache.
sl@0
    16
// 
sl@0
    17
//
sl@0
    18
sl@0
    19
#include <e32test.h>
sl@0
    20
#include <f32file.h>
sl@0
    21
#include <hal.h>
sl@0
    22
#include <bautils.h>
sl@0
    23
#include <babitflags.h>
sl@0
    24
#include <babackup.h>
sl@0
    25
#include <startup.hrh>
sl@0
    26
#include <ecom/resolver.h>
sl@0
    27
#include "EComErrorCodes.h"
sl@0
    28
#include "EComUidCodes.h"
sl@0
    29
#include "ImplementationInformation.h"
sl@0
    30
#include "RegistryData.h"
sl@0
    31
#include "Registrar.h"
sl@0
    32
#include "RegistrarObserver.h"
sl@0
    33
#include "../EcomTestUtils/EcomTestUtils.h"
sl@0
    34
#include <swi/swispubsubdefs.h>
sl@0
    35
#include "../EcomTestUtils/TPropertyManager.h"
sl@0
    36
#include "EComPatchDataConstantv2.h"
sl@0
    37
#include "RegistryResolveTransaction.h"  // ecom3 code
sl@0
    38
#include "callback.h"
sl@0
    39
#include "resolvercache.h"
sl@0
    40
#include "EComServer.h"
sl@0
    41
sl@0
    42
sl@0
    43
const TInt KOneSecond = 1000000;
sl@0
    44
const TInt KHalfSecond = KOneSecond / 2;
sl@0
    45
// Use this timeout to wait for events occuring within a few seconds.
sl@0
    46
const TInt KIndefiniteWait = KOneSecond * 20;
sl@0
    47
sl@0
    48
LOCAL_D RTest test(_L("t_resolvercache.exe"));
sl@0
    49
sl@0
    50
LOCAL_D RFs					TheFs;
sl@0
    51
LOCAL_D CTrapCleanup* 		TheTrapCleanup 		= NULL;
sl@0
    52
class CDerivedActiveScheduler;
sl@0
    53
LOCAL_D CDerivedActiveScheduler* TheActiveScheduler	= NULL;
sl@0
    54
sl@0
    55
sl@0
    56
// custom resolvers available for testing.
sl@0
    57
// 200126cd, A0001346 and A0001347 are allocated outside the
sl@0
    58
// ECOM Uid Allocations.doc
sl@0
    59
const TUid KDummyResolverUid2  = {0xA0001346};
sl@0
    60
const TUid KDummyResolverUid3  = {0xA0001347};
sl@0
    61
const TUid KExampleResolverUid = {0x10009DD0};
sl@0
    62
const TUid KMyResolverUid      = {0x10009E12};
sl@0
    63
const TUid KDummyResolverUid1  = {0x200126CD};
sl@0
    64
sl@0
    65
// The custom resolver in RAMOnly dir
sl@0
    66
_LIT(KDummyRscInC, "c:\\resource\\plugins\\dummycustomresolver1.rsc");
sl@0
    67
_LIT(KDummyDllInC, "c:\\sys\\bin\\dummycustomresolver1.dll");
sl@0
    68
sl@0
    69
_LIT(KDummyRscInZ, "z:\\ramonly\\dummycustomresolver1.rsc");
sl@0
    70
_LIT(KDummyDllInZ, "z:\\ramonly\\dummycustomresolver1.dll");
sl@0
    71
sl@0
    72
// This pair is to upgrade a resolver DLL.
sl@0
    73
// NB: to supersede a Z: drive plugin, the C: DLL must have the
sl@0
    74
// same name as the one in z: drive. Hence the '2' is dropped.
sl@0
    75
_LIT(KDllUpgradeInZ, "z:\\ramonly\\cachedcustomresolver2.dll");
sl@0
    76
_LIT(KDllUpgradeInC, "c:\\sys\\bin\\cachedcustomresolver.dll");
sl@0
    77
sl@0
    78
_LIT(KRscUpgradeInZ, "z:\\ramonly\\cachedcustomresolver2.rsc");
sl@0
    79
_LIT(KRscUpgradeInC, "c:\\resource\\plugins\\cachedcustomresolver.rsc");
sl@0
    80
sl@0
    81
/** User::AfterHighRes is not a reliable mechanism to wait
sl@0
    82
for async events. Especially when we have 4 different timers
sl@0
    83
firing within a span of 4 s. Hence this harness intercepts
sl@0
    84
and inserts callbacks in the notification sources and set
sl@0
    85
the following flag to indicate what event has occured. */
sl@0
    86
LOCAL_D TBitFlags32 AsyncEvents = 0;
sl@0
    87
sl@0
    88
/** enum to identify different async events */
sl@0
    89
enum TAsyncEventId
sl@0
    90
	{
sl@0
    91
	EAsyncEvent_Unknown = 0,
sl@0
    92
	EAsyncEvent_ImplUpgrade,
sl@0
    93
	EAsyncEvent_SwiStart,
sl@0
    94
	EAsyncEvent_SwiEnd,
sl@0
    95
	EAsyncEvent_BurStart,
sl@0
    96
	EAsyncEvent_BurEnd,
sl@0
    97
	EAsyncEvent_CacheTimer,
sl@0
    98
	EAsyncEvent_HaltTimer
sl@0
    99
	};
sl@0
   100
sl@0
   101
//
sl@0
   102
// maps callback data to enum TAsyncEventId.
sl@0
   103
LOCAL_C TAsyncEventId CallbackDataToEventId(TInt aEvent, TAny* aData)
sl@0
   104
	{
sl@0
   105
	TAsyncEventId ret = EAsyncEvent_Unknown;
sl@0
   106
	TCallBackState* state = static_cast<TCallBackState*>(aData);
sl@0
   107
	if (aEvent == ECallBackId_SwiEvent)
sl@0
   108
		{
sl@0
   109
		if (*state == ECallBackState_EventStart)
sl@0
   110
			{
sl@0
   111
			ret = EAsyncEvent_SwiStart;
sl@0
   112
			}
sl@0
   113
		else // treat all unexpected states as SWI end!
sl@0
   114
			{
sl@0
   115
			ret = EAsyncEvent_SwiEnd;
sl@0
   116
			}
sl@0
   117
		}
sl@0
   118
	else if (aEvent == ECallBackId_BurEvent)
sl@0
   119
		{
sl@0
   120
		if (*state == ECallBackState_EventStart)
sl@0
   121
			{
sl@0
   122
			ret = EAsyncEvent_BurStart;
sl@0
   123
			}
sl@0
   124
		else // treat all unexpected states as BUR finish!
sl@0
   125
			{
sl@0
   126
			ret = EAsyncEvent_BurEnd;
sl@0
   127
			}
sl@0
   128
		}
sl@0
   129
	else if (aEvent == ECallBackId_ImplUpgrade)
sl@0
   130
		{
sl@0
   131
		ret = EAsyncEvent_ImplUpgrade;
sl@0
   132
		}
sl@0
   133
	return ret;
sl@0
   134
	}
sl@0
   135
sl@0
   136
// beginningOfTest is set when this harness is run.
sl@0
   137
LOCAL_D TTime beginningOfTest;
sl@0
   138
LOCAL_C void WaitForLazyUnloadingL()
sl@0
   139
	{
sl@0
   140
	TTime now;
sl@0
   141
	now.UniversalTime();
sl@0
   142
sl@0
   143
	TTimeIntervalSeconds n;
sl@0
   144
	User::LeaveIfError(now.SecondsFrom(beginningOfTest, n));
sl@0
   145
	const TInt KLazyDllUnloadPeriod =  150; // actual is 2 minutes
sl@0
   146
	TInt secondsToWait = KLazyDllUnloadPeriod - n.Int();
sl@0
   147
	test.Printf(_L("Amount to wait for lazy unload is %d seconds.\n"), secondsToWait);
sl@0
   148
	if (secondsToWait > 0)
sl@0
   149
		{
sl@0
   150
		User::After(KOneSecond * secondsToWait);
sl@0
   151
		}
sl@0
   152
	}
sl@0
   153
sl@0
   154
// Copies the Plugins to specific folder for testing purpose
sl@0
   155
LOCAL_C void CopyPluginsL()
sl@0
   156
	{
sl@0
   157
	EComTestUtils::FileManCopyFileL(KDummyRscInZ, KDummyRscInC);
sl@0
   158
	EComTestUtils::FileManCopyFileL(KDummyDllInZ, KDummyDllInC);
sl@0
   159
	}
sl@0
   160
sl@0
   161
// Deleting plugin from the RAM for cleanup purpose
sl@0
   162
LOCAL_C void DeleteTestPlugin()
sl@0
   163
	{
sl@0
   164
	TRAP_IGNORE(EComTestUtils::FileManDeleteFileL(KDummyRscInC));
sl@0
   165
	TRAP_IGNORE(EComTestUtils::FileManDeleteFileL(KDummyDllInC));
sl@0
   166
sl@0
   167
	TRAP_IGNORE(EComTestUtils::FileManDeleteFileL(KRscUpgradeInC));
sl@0
   168
	TRAP_IGNORE(EComTestUtils::FileManDeleteFileL(KDllUpgradeInC));
sl@0
   169
	}
sl@0
   170
sl@0
   171
// utility to help cast TTimeIntervalMicroSeconds to TInt
sl@0
   172
LOCAL_C TInt CalcElapsedMicroseconds(const TTime& aStart)
sl@0
   173
	{
sl@0
   174
	TTime now;
sl@0
   175
	now.UniversalTime();
sl@0
   176
	TTimeIntervalMicroSeconds timediff = now.MicroSecondsFrom(aStart);
sl@0
   177
	return I64LOW( timediff.Int64() );
sl@0
   178
	}
sl@0
   179
sl@0
   180
/** Need a CActive to wait for various change notifications. */
sl@0
   181
class CHaltTimer : public CTimer
sl@0
   182
	{
sl@0
   183
public:
sl@0
   184
	CHaltTimer(TInt aPriority);
sl@0
   185
	~CHaltTimer();
sl@0
   186
	void ConstructL();
sl@0
   187
	void StartTimer(TInt aTimeInterval);
sl@0
   188
sl@0
   189
private:
sl@0
   190
	void RunL();
sl@0
   191
	};
sl@0
   192
sl@0
   193
CHaltTimer::CHaltTimer(TInt aPriority)
sl@0
   194
	: CTimer(aPriority)
sl@0
   195
	{
sl@0
   196
	CActiveScheduler::Add(this);
sl@0
   197
	}
sl@0
   198
sl@0
   199
CHaltTimer::~CHaltTimer()
sl@0
   200
	{
sl@0
   201
	Cancel();
sl@0
   202
	}
sl@0
   203
sl@0
   204
void CHaltTimer::ConstructL()
sl@0
   205
	{
sl@0
   206
	CTimer::ConstructL();
sl@0
   207
	}
sl@0
   208
sl@0
   209
void CHaltTimer::StartTimer(TInt aTimeInterval)
sl@0
   210
	{
sl@0
   211
	After(aTimeInterval);
sl@0
   212
	}
sl@0
   213
sl@0
   214
void CHaltTimer::RunL()
sl@0
   215
	{
sl@0
   216
	AsyncEvents.Set(EAsyncEvent_HaltTimer);
sl@0
   217
	CActiveScheduler::Stop();
sl@0
   218
	}
sl@0
   219
sl@0
   220
/** Avoid E32User::Case 47 panic in OOM test */
sl@0
   221
class CDerivedActiveScheduler : public CActiveScheduler
sl@0
   222
	{
sl@0
   223
public:
sl@0
   224
	virtual void Error(TInt aError) const;
sl@0
   225
	};
sl@0
   226
sl@0
   227
void CDerivedActiveScheduler::Error(TInt aError) const
sl@0
   228
	{
sl@0
   229
	Halt(aError);
sl@0
   230
	}
sl@0
   231
sl@0
   232
/** friend class to access private members of CEComServer */
sl@0
   233
class TEComServer_StateAccessor
sl@0
   234
	{
sl@0
   235
public:
sl@0
   236
	static void InterceptCallbacks(CEComServer& aEComServer, TCallBackWithArg aCb);
sl@0
   237
	static CCustomResolverCache* GetResolverCache(CEComServer& aEComServer);
sl@0
   238
	};
sl@0
   239
sl@0
   240
/** Test class for object CCustomResolverCache.
sl@0
   241
*/
sl@0
   242
class CCustomResolverCacheTest : public CBase
sl@0
   243
	{
sl@0
   244
public:
sl@0
   245
	typedef void (CCustomResolverCacheTest::*ClassFuncPtrL) (void);
sl@0
   246
sl@0
   247
	virtual ~CCustomResolverCacheTest();
sl@0
   248
	static CCustomResolverCacheTest* NewL();
sl@0
   249
	static void RunAllTestsL();
sl@0
   250
	static TInt InterceptCacheMgrCallback(TAny* aObj, TInt aEvent, TAny* aData);
sl@0
   251
	static TInt CacheTimeoutCallback(TAny* aObj, TInt aEvent, TAny* aData);
sl@0
   252
sl@0
   253
private:
sl@0
   254
	CCustomResolverCacheTest();
sl@0
   255
	void ConstructL();
sl@0
   256
sl@0
   257
	static void DoBasicTestL(ClassFuncPtrL aTestFunc);
sl@0
   258
	static void DoOOMTestL(ClassFuncPtrL aTestFunc);
sl@0
   259
sl@0
   260
	// Test cases
sl@0
   261
	void TestUpgradingCachedResolverL();
sl@0
   262
	void TestDeletingCachedResolverL();
sl@0
   263
sl@0
   264
	void TestCacheQueueFullPattern1L();
sl@0
   265
	void TestCacheQueueFullPattern2L();
sl@0
   266
	void TestCacheQueueFullPattern3L();
sl@0
   267
	void DoQueueFullTestL(const RArray<TUid>& aResolverList);
sl@0
   268
sl@0
   269
	void TestCounterWrapAroundL();
sl@0
   270
	void TestCacheTimerAccuracyL();
sl@0
   271
	void TestTimestampUpdateOnCacheHitL();
sl@0
   272
	void TestSWIDisableRwResolverCachingL();
sl@0
   273
	void TestBurDisableRwResolverCachingL();
sl@0
   274
	void TestClockChangeHasNoEffectOnCacheTimeoutL();
sl@0
   275
	void TestCacheSizeZeroL();
sl@0
   276
	void TestCacheTimeoutZeroL();
sl@0
   277
	void TestResolverWithBadProxyTable();
sl@0
   278
sl@0
   279
	// utilities
sl@0
   280
	TBool UseResolverCheckVersionL(TUid aResolverUid,
sl@0
   281
								   TInt aVersion = 0);
sl@0
   282
	void YieldToOtherCActive(TInt aMicroSeconds);
sl@0
   283
	void WaitAsyncL(TInt aNumSeconds);
sl@0
   284
	TBool WaitForEvict(TUid aResolverUid);
sl@0
   285
sl@0
   286
	// access private data of CCustomResolverCache
sl@0
   287
	TInt CacheSize();
sl@0
   288
	TBool HasResolverUid(TUid aUid);
sl@0
   289
	TBool QueueIsSorted();
sl@0
   290
	inline CCustomResolverCache* ResolverCache();
sl@0
   291
sl@0
   292
private:
sl@0
   293
	/** need a CActive to wait for other async events */
sl@0
   294
	CHaltTimer* iHaltTimer;
sl@0
   295
sl@0
   296
	CEComServer* iEComServer;
sl@0
   297
	};
sl@0
   298
sl@0
   299
//==============================================
sl@0
   300
// class TEComServer_StateAccessor
sl@0
   301
//==============================================
sl@0
   302
void TEComServer_StateAccessor::InterceptCallbacks(CEComServer& aEComServer,
sl@0
   303
												   TCallBackWithArg aCb)
sl@0
   304
	{
sl@0
   305
	aEComServer.iRegistrar->InstallSwiEventCallBack(aCb);
sl@0
   306
	aEComServer.iRegistrar->InstallBurEventCallBack(aCb);
sl@0
   307
	aEComServer.iRegistryData->SetImplUpgradeCallBack(aCb);
sl@0
   308
	}
sl@0
   309
sl@0
   310
CCustomResolverCache*
sl@0
   311
TEComServer_StateAccessor::GetResolverCache(CEComServer& aEComServer)
sl@0
   312
	{
sl@0
   313
	return aEComServer.iResolverCache;
sl@0
   314
	}
sl@0
   315
sl@0
   316
//==============================================
sl@0
   317
// class CCustomResolverCacheTest
sl@0
   318
//==============================================
sl@0
   319
sl@0
   320
/** CCustomResolverCacheTest constructor */
sl@0
   321
CCustomResolverCacheTest::CCustomResolverCacheTest()
sl@0
   322
	{
sl@0
   323
	}
sl@0
   324
sl@0
   325
/** CCustomResolverCacheTest destructor. */
sl@0
   326
CCustomResolverCacheTest::~CCustomResolverCacheTest()
sl@0
   327
	{
sl@0
   328
	delete iEComServer;
sl@0
   329
	delete iHaltTimer;
sl@0
   330
	}
sl@0
   331
sl@0
   332
/** static factory method to instantiate CCustomResolverCacheTest object.
sl@0
   333
*/
sl@0
   334
CCustomResolverCacheTest* CCustomResolverCacheTest::NewL()
sl@0
   335
	{
sl@0
   336
	CCustomResolverCacheTest* self = new(ELeave) CCustomResolverCacheTest;
sl@0
   337
	CleanupStack::PushL(self);
sl@0
   338
	self->ConstructL();
sl@0
   339
	CleanupStack::Pop(self);
sl@0
   340
	return self;
sl@0
   341
	}
sl@0
   342
sl@0
   343
/**
sl@0
   344
Standardized 2nd phase of two phase construction.
sl@0
   345
*/
sl@0
   346
void CCustomResolverCacheTest::ConstructL()
sl@0
   347
	{
sl@0
   348
	iHaltTimer = new(ELeave) CHaltTimer(CActive::EPriorityIdle);
sl@0
   349
	iHaltTimer->ConstructL();
sl@0
   350
sl@0
   351
	iEComServer = CEComServer::NewLC();
sl@0
   352
	CleanupStack::Pop(iEComServer);
sl@0
   353
sl@0
   354
	TCallBackWithArg interceptorCB(&InterceptCacheMgrCallback, this);
sl@0
   355
	TEComServer_StateAccessor::InterceptCallbacks(*iEComServer, interceptorCB);
sl@0
   356
sl@0
   357
	TCallBackWithArg cacheTimerCb(&CacheTimeoutCallback, this);
sl@0
   358
	ResolverCache()->iTimerExpireCB = cacheTimerCb;
sl@0
   359
	}
sl@0
   360
sl@0
   361
/** util to fetch CEComServer::iResolverCache */
sl@0
   362
inline CCustomResolverCache* CCustomResolverCacheTest::ResolverCache()
sl@0
   363
	{
sl@0
   364
	return TEComServer_StateAccessor::GetResolverCache(*iEComServer);
sl@0
   365
	}
sl@0
   366
sl@0
   367
/** the test harness install this callback with ECOM server objects
sl@0
   368
to intercept async events. This callback will relay the original
sl@0
   369
call to CEComServer, then raise a flag to indicate what
sl@0
   370
event has occurred.
sl@0
   371
@param aObj pointer to the CCustomResolverCacheTest object
sl@0
   372
@param aEvent ID of event triggering the callback
sl@0
   373
@param aData  pointer to some data associated with the event.
sl@0
   374
@return Always KErrNone. It is ignored.
sl@0
   375
*/
sl@0
   376
TInt CCustomResolverCacheTest::InterceptCacheMgrCallback(TAny* aObj,
sl@0
   377
														 TInt aEvent,
sl@0
   378
														 TAny* aData)
sl@0
   379
	{
sl@0
   380
	CCustomResolverCacheTest* self = static_cast<CCustomResolverCacheTest*>(aObj);
sl@0
   381
	// Pass the event along to let CEComServer does its thing.
sl@0
   382
	CEComServer::NotifyEvents(self->iEComServer, aEvent, aData);
sl@0
   383
sl@0
   384
	if (self->iHaltTimer->IsActive())
sl@0
   385
		{
sl@0
   386
		TAsyncEventId event = CallbackDataToEventId(aEvent, aData);
sl@0
   387
		AsyncEvents.Set(event);
sl@0
   388
sl@0
   389
		self->iHaltTimer->Cancel();
sl@0
   390
		CActiveScheduler::Stop();
sl@0
   391
		}
sl@0
   392
	else
sl@0
   393
		{
sl@0
   394
		// BUR and SWI unit test cases just call the notifier
sl@0
   395
		// directly. So CHaltTimer is not running.
sl@0
   396
		test.Printf(_L("CacheTest: caught async event %d when timer not running\n"), aEvent);
sl@0
   397
		}
sl@0
   398
	return KErrNone;
sl@0
   399
	}
sl@0
   400
sl@0
   401
/** A callback installed in cache mgr. Whenever the cache timer expires
sl@0
   402
this callback is invoked by CCustomResolverCache
sl@0
   403
@param aObj pointer to CCustomResolverCacheTest
sl@0
   404
@return Always KErrNone. It is ignored.
sl@0
   405
*/
sl@0
   406
TInt CCustomResolverCacheTest::CacheTimeoutCallback(TAny* aObj, TInt, TAny*)
sl@0
   407
	{
sl@0
   408
	AsyncEvents.Set(EAsyncEvent_CacheTimer);
sl@0
   409
sl@0
   410
	CCustomResolverCacheTest* self = static_cast<CCustomResolverCacheTest*>(aObj);
sl@0
   411
	self->iHaltTimer->Cancel();
sl@0
   412
	CActiveScheduler::Stop();
sl@0
   413
	return KErrNone;
sl@0
   414
	}
sl@0
   415
sl@0
   416
/** Wrapper function to run normal mode (non-OOM) test
sl@0
   417
*/
sl@0
   418
void CCustomResolverCacheTest::DoBasicTestL(ClassFuncPtrL aTestFunc)
sl@0
   419
	{
sl@0
   420
	__UHEAP_MARK;
sl@0
   421
	TInt startProcessHandleCount;
sl@0
   422
	TInt startThreadHandleCount;
sl@0
   423
	RThread().HandleCount(startProcessHandleCount, startThreadHandleCount);
sl@0
   424
sl@0
   425
	CCustomResolverCacheTest* p = CCustomResolverCacheTest::NewL();
sl@0
   426
	(p->*aTestFunc)();
sl@0
   427
	delete p;
sl@0
   428
sl@0
   429
	// check that no handles have leaked
sl@0
   430
	TInt endProcessHandleCount;
sl@0
   431
	TInt endThreadHandleCount;
sl@0
   432
	RThread().HandleCount(endProcessHandleCount, endThreadHandleCount);
sl@0
   433
sl@0
   434
	test(startProcessHandleCount == endProcessHandleCount);
sl@0
   435
	test(startThreadHandleCount  == endThreadHandleCount);
sl@0
   436
	__UHEAP_MARKEND;
sl@0
   437
	}
sl@0
   438
sl@0
   439
/**
sl@0
   440
@SYMTestCaseID		SYSLIB-ECOM-UT-4020
sl@0
   441
@SYMTestCaseDesc 	Verify no memory leak in CCustomResolverCache.
sl@0
   442
@SYMTestPriority 	Critical
sl@0
   443
@SYMTestActions  	Run UT-4015, UT-4016 and UT-4017 under OOM.
sl@0
   444
@SYMTestExpectedResults No memory leak, no handle leak.
sl@0
   445
@SYMCR CR1182
sl@0
   446
*/
sl@0
   447
void CCustomResolverCacheTest::DoOOMTestL(ClassFuncPtrL aTestFunc)
sl@0
   448
	{
sl@0
   449
	test.Next(_L(" @SYMTestCaseID:SYSLIB-ECOM-UT-4020 "));
sl@0
   450
	__UHEAP_MARK;
sl@0
   451
	TInt startProcessHandleCount;
sl@0
   452
	TInt startThreadHandleCount;
sl@0
   453
	RThread().HandleCount(startProcessHandleCount, startThreadHandleCount);
sl@0
   454
	TInt err(KErrNone);
sl@0
   455
	TInt oomStep = 0;
sl@0
   456
sl@0
   457
	do {
sl@0
   458
		// Instantiation of CCustomResolverCacheTest involves building
sl@0
   459
		// the registry. It will take couple hours under OOM. Hence do it
sl@0
   460
		// outside __UHEAP_SETFAIL.
sl@0
   461
		CCustomResolverCacheTest* p = CCustomResolverCacheTest::NewL();
sl@0
   462
sl@0
   463
		__UHEAP_SETFAIL(RHeap::EDeterministic, ++oomStep);
sl@0
   464
		TRAP(err, (p->*aTestFunc)());
sl@0
   465
		__UHEAP_SETFAIL(RHeap::ENone, 0);
sl@0
   466
sl@0
   467
		delete p;
sl@0
   468
	} while (err == KErrNoMemory);
sl@0
   469
sl@0
   470
	test(err == KErrNone);
sl@0
   471
	test.Printf(_L("OOM succeeded at heap failure rate %d\n"), oomStep);
sl@0
   472
sl@0
   473
	// check that no handles have leaked
sl@0
   474
	TInt endProcessHandleCount;
sl@0
   475
	TInt endThreadHandleCount;
sl@0
   476
	RThread().HandleCount(endProcessHandleCount, endThreadHandleCount);
sl@0
   477
sl@0
   478
	test(startProcessHandleCount == endProcessHandleCount);
sl@0
   479
	test(startThreadHandleCount  == endThreadHandleCount);
sl@0
   480
	__UHEAP_MARKEND;
sl@0
   481
	}
sl@0
   482
sl@0
   483
/** helper, ecom3 code
sl@0
   484
Issue a list request to add resolver to cache and check if
sl@0
   485
the registry has the expected version of the resolver.
sl@0
   486
@param aResolverUid the custom resolver to create
sl@0
   487
@param aVersion the expected version of the custom resolver. Check is skipped
sl@0
   488
  if aVersion is zero.
sl@0
   489
*/
sl@0
   490
TBool CCustomResolverCacheTest::UseResolverCheckVersionL(TUid aResolverUid,
sl@0
   491
														TInt aVersion)
sl@0
   492
	{
sl@0
   493
	TEComResolverParams resolverparams;
sl@0
   494
	_LIT8(KDummyData,"dummy");
sl@0
   495
	resolverparams.SetDataType(KDummyData);
sl@0
   496
sl@0
   497
	TClientRequest clientReq;
sl@0
   498
	RArray<TUid> extendedInterfaces;
sl@0
   499
	RImplInfoArray* infoArray = iEComServer->ListImplementationsL(
sl@0
   500
		KCustomResolverInterfaceUid, resolverparams, aResolverUid,
sl@0
   501
		extendedInterfaces, clientReq);
sl@0
   502
sl@0
   503
	TBool ret = (aVersion == 0);
sl@0
   504
sl@0
   505
	// infoArray not pushed because there are no leave in this search.
sl@0
   506
	for (TInt i = 0; !ret && i < infoArray->Count(); i++)
sl@0
   507
		{
sl@0
   508
		const CImplementationInformation* impl = (*infoArray)[i];
sl@0
   509
		if (impl->ImplementationUid() == aResolverUid && impl->Version() == aVersion)
sl@0
   510
			{
sl@0
   511
			ret = ETrue;
sl@0
   512
			}
sl@0
   513
		}
sl@0
   514
sl@0
   515
	infoArray->Reset();
sl@0
   516
	delete infoArray;
sl@0
   517
sl@0
   518
	return ret;
sl@0
   519
	}
sl@0
   520
sl@0
   521
/** utility to let other CActive s to run. */
sl@0
   522
void CCustomResolverCacheTest::YieldToOtherCActive(TInt aMicroSeconds)
sl@0
   523
	{
sl@0
   524
	iHaltTimer->StartTimer(aMicroSeconds);
sl@0
   525
	CActiveScheduler::Start();
sl@0
   526
	}
sl@0
   527
sl@0
   528
/** Call YieldToOtherCActive as many times as needed
sl@0
   529
until aNumSeconds is reached. */
sl@0
   530
void CCustomResolverCacheTest::WaitAsyncL(TInt aNumSeconds)
sl@0
   531
	{
sl@0
   532
	TTime start, now;
sl@0
   533
	start.UniversalTime();
sl@0
   534
	TTimeIntervalSeconds elapsed;
sl@0
   535
sl@0
   536
	for (elapsed = 0; elapsed.Int() < aNumSeconds; )
sl@0
   537
		{
sl@0
   538
		AsyncEvents.ClearAll();
sl@0
   539
		TInt seconds = (aNumSeconds - elapsed.Int());
sl@0
   540
		YieldToOtherCActive(KOneSecond * seconds);
sl@0
   541
		now.UniversalTime();
sl@0
   542
		User::LeaveIfError(now.SecondsFrom(start, elapsed));
sl@0
   543
		}
sl@0
   544
	}
sl@0
   545
sl@0
   546
/**
sl@0
   547
This method is used in the situation that cache timer is already
sl@0
   548
started when we load the given resolver. Then when cache timer
sl@0
   549
expires, the resolver is not ripe to be evicted. Have to wait a
sl@0
   550
second time.
sl@0
   551
@param aResolverUid the Resolver to wait for evict.
sl@0
   552
@return ETrue means the resolver is evicted correctly.
sl@0
   553
		EFalse means error, i.e. test fail.
sl@0
   554
*/
sl@0
   555
TBool CCustomResolverCacheTest::WaitForEvict(TUid aResolverUid)
sl@0
   556
	{
sl@0
   557
	TBool ret = EFalse;
sl@0
   558
	// No point of waiting for 20s for cache timeout.
sl@0
   559
	TInt waitMicroSec = KCustomResolverCacheTimeout + KOneSecond;
sl@0
   560
	for (TInt i = 0; (i < 2) && !ret; i++)
sl@0
   561
		{
sl@0
   562
		AsyncEvents.ClearAll();
sl@0
   563
		YieldToOtherCActive(waitMicroSec);
sl@0
   564
		// Check which async event has stopped the activescheduler
sl@0
   565
		test( AsyncEvents.IsSet(EAsyncEvent_CacheTimer) );
sl@0
   566
		ret = ! HasResolverUid(aResolverUid);
sl@0
   567
		}
sl@0
   568
	return ret;
sl@0
   569
	}
sl@0
   570
sl@0
   571
//==========================================================
sl@0
   572
// utilities to access internal data of CCustomResolverCache.
sl@0
   573
// This is possible because CCustomResolverCacheTest is a friend.
sl@0
   574
//===========================================================
sl@0
   575
sl@0
   576
TInt CCustomResolverCacheTest::CacheSize()
sl@0
   577
	{
sl@0
   578
	return ResolverCache()->iResolvers.Count();
sl@0
   579
	}
sl@0
   580
sl@0
   581
TBool CCustomResolverCacheTest::HasResolverUid(TUid aUid)
sl@0
   582
	{
sl@0
   583
	for (TInt i = 0; i < ResolverCache()->iResolvers.Count(); i++)
sl@0
   584
		{
sl@0
   585
		if (ResolverCache()->iResolvers[i].iResolverUid == aUid)
sl@0
   586
			{
sl@0
   587
			return ETrue;
sl@0
   588
			}
sl@0
   589
		}
sl@0
   590
    return EFalse;
sl@0
   591
	}
sl@0
   592
sl@0
   593
TBool CCustomResolverCacheTest::QueueIsSorted()
sl@0
   594
	{
sl@0
   595
	CCustomResolverCache* theResolverCache = ResolverCache();
sl@0
   596
	for (TInt i = 1; i < theResolverCache->iResolvers.Count(); i++)
sl@0
   597
		{
sl@0
   598
		if (theResolverCache->iResolvers[i-1].iResolverUid.iUid >=
sl@0
   599
			theResolverCache->iResolvers[i].iResolverUid.iUid)
sl@0
   600
			{
sl@0
   601
			test.Printf(_L("Sort error: i-1 %d, 0x%8X, 0x%8X\n"), i-1,
sl@0
   602
				theResolverCache->iResolvers[i-1].iResolverUid.iUid,
sl@0
   603
				theResolverCache->iResolvers[i].iResolverUid.iUid	);
sl@0
   604
			return EFalse;
sl@0
   605
			}
sl@0
   606
		}
sl@0
   607
    return ETrue;
sl@0
   608
	}
sl@0
   609
sl@0
   610
/**
sl@0
   611
@SYMTestCaseID		SYSLIB-ECOM-CT-4012
sl@0
   612
@SYMTestCaseDesc 	If a cached resolver is superseded by a new version, it
sl@0
   613
	is evicted from cache.
sl@0
   614
@SYMTestPriority 	High
sl@0
   615
@SYMTestActions  	1. Use a custom resolver to get it in cache.
sl@0
   616
	2. copy the upgrade version of the resolver to C: and wait for
sl@0
   617
	   its discovery.
sl@0
   618
@SYMTestExpectedResults 1. a/. The resolver is the base version.
sl@0
   619
       b/. It is added in the cache.
sl@0
   620
	2. a/. Cache mgr receives upgrade notification (from CRegistryData).
sl@0
   621
	   b/. The resolver entry disappears from cache and the cache timer is
sl@0
   622
	       still running.
sl@0
   623
@SYMCR CR1182
sl@0
   624
*/
sl@0
   625
void CCustomResolverCacheTest::TestUpgradingCachedResolverL()
sl@0
   626
	{
sl@0
   627
	test.Next(_L(" @SYMTestCaseID:SYSLIB-ECOM-CT-4012 "));
sl@0
   628
	// get the customer resolver in cache. It should be version 1.
sl@0
   629
	test (UseResolverCheckVersionL(KMyResolverUid, 1));
sl@0
   630
	// Check that KMyResolverUid is added to cache.
sl@0
   631
	test(HasResolverUid(KMyResolverUid));
sl@0
   632
sl@0
   633
	// Need an extra item in cache to get Bullseye to check off
sl@0
   634
	// a conditional. It is not an active ingredient of this test.
sl@0
   635
	UseResolverCheckVersionL(KDummyResolverUid1, 0);
sl@0
   636
sl@0
   637
	// Copy the upgrade to C: drive
sl@0
   638
	EComTestUtils::FileManCopyFileL(KRscUpgradeInZ, KRscUpgradeInC);
sl@0
   639
	EComTestUtils::FileManCopyFileL(KDllUpgradeInZ, KDllUpgradeInC);
sl@0
   640
sl@0
   641
	// Let CDiscoverer discover the resource file.
sl@0
   642
	AsyncEvents.ClearAll();
sl@0
   643
	YieldToOtherCActive(KIndefiniteWait);  // 20 s
sl@0
   644
sl@0
   645
	// Check which async event has stopped the activescheduler
sl@0
   646
	test( AsyncEvents.IsSet(EAsyncEvent_ImplUpgrade) );
sl@0
   647
sl@0
   648
	// And the cache entry should be gone.
sl@0
   649
	test(! HasResolverUid(KMyResolverUid));
sl@0
   650
sl@0
   651
	// cleanup. NB: we never loaded the DLL in C: drive so
sl@0
   652
	// no need to worry about lazy dll unloading.
sl@0
   653
	EComTestUtils::FileManDeleteFileL(KRscUpgradeInC);
sl@0
   654
	EComTestUtils::FileManDeleteFileL(KDllUpgradeInC);
sl@0
   655
	}
sl@0
   656
sl@0
   657
/**
sl@0
   658
@SYMTestCaseID		SYSLIB-ECOM-CT-4013
sl@0
   659
@SYMTestCaseDesc 	Verify a cached DLL cannot be deleted until it
sl@0
   660
	is removed from cache. This test serves as a sanity check that
sl@0
   661
	ECOM does need these notifications to trigger unloading DLL
sl@0
   662
	on SWI and BUR events.
sl@0
   663
@SYMTestPriority 	Medium
sl@0
   664
@SYMTestActions  	1. Use a RW drive resolver to get it in cache.
sl@0
   665
	2. Try to delete the DLL within default cache timeout period.
sl@0
   666
    3. Try again after it is evicted from cache.
sl@0
   667
@SYMTestExpectedResults 1. The resolver is added in cache.
sl@0
   668
	2. Get KErrAccessDenied error.
sl@0
   669
	3. The delete is either KErrNone or KErrNotFound because in step 2
sl@0
   670
	   even though the return code is -21, the file may be actually gone.
sl@0
   671
@SYMCR CR1182
sl@0
   672
*/
sl@0
   673
void CCustomResolverCacheTest::TestDeletingCachedResolverL()
sl@0
   674
	{
sl@0
   675
	test.Next(_L(" @SYMTestCaseID:SYSLIB-ECOM-CT-4013 "));
sl@0
   676
	// NB: lazy Dll unloading interferes with this test.
sl@0
   677
	// Best to run this test at the end.
sl@0
   678
	WaitForLazyUnloadingL();
sl@0
   679
sl@0
   680
	// There is only 1 version of 200126CD. No need to check version.
sl@0
   681
	// So ignore the return code.
sl@0
   682
	UseResolverCheckVersionL(KDummyResolverUid1, 0);
sl@0
   683
sl@0
   684
	// Check that it is added to cache.
sl@0
   685
	test( HasResolverUid(KDummyResolverUid1) );
sl@0
   686
sl@0
   687
	TInt err(KErrNone);
sl@0
   688
	TRAP(err, EComTestUtils::FileManDeleteFileL(KDummyDllInC));
sl@0
   689
sl@0
   690
#ifndef __WINSCW__
sl@0
   691
	// On hw you can delete the DLL while it is loaded.
sl@0
   692
	if (err == KErrNone)
sl@0
   693
		{
sl@0
   694
		test.Printf(_L("Delete test: RFs allows deletion of loaded DLL on hw. Test not run.\n"));
sl@0
   695
		EComTestUtils::FileManCopyFileL(KDummyDllInZ, KDummyDllInC);
sl@0
   696
		return;
sl@0
   697
		}
sl@0
   698
#endif
sl@0
   699
sl@0
   700
	test(err == KErrAccessDenied);
sl@0
   701
sl@0
   702
	// Wait for its eviction after cache timeout.
sl@0
   703
	AsyncEvents.ClearAll();
sl@0
   704
	YieldToOtherCActive(KIndefiniteWait);  // 20 s
sl@0
   705
sl@0
   706
	// Check which async event has stopped the activescheduler
sl@0
   707
	test( AsyncEvents.IsSet(EAsyncEvent_CacheTimer) );
sl@0
   708
	// Check the resolver is evicted
sl@0
   709
	test( !HasResolverUid(KDummyResolverUid1) );
sl@0
   710
sl@0
   711
	TRAP(err, EComTestUtils::FileManDeleteFileL(KDummyDllInC));
sl@0
   712
	test(err == KErrNone || err == KErrNotFound);
sl@0
   713
sl@0
   714
	// restore the file
sl@0
   715
	EComTestUtils::FileManCopyFileL(KDummyDllInZ, KDummyDllInC);
sl@0
   716
	}
sl@0
   717
sl@0
   718
/**
sl@0
   719
@SYMTestCaseID		SYSLIB-ECOM-UT-4015
sl@0
   720
@SYMTestCaseDesc 	Verify cache queue does not grow beyond the max
sl@0
   721
	queue size. Verify when a resolver is added to a full cache,
sl@0
   722
	the one to evict is the least recently used.
sl@0
   723
sl@0
   724
	There are 3 versions of this test. In Pattern1, the LRU entry
sl@0
   725
	has UID value which is the smallest of the 5 resolvers.
sl@0
   726
	In pattern 2, UID of the LRU is greatest. In pattern 3, UID of the
sl@0
   727
	LRU is the second smallest.
sl@0
   728
@SYMTestPriority 	High
sl@0
   729
@SYMTestActions  	1. Use 4 different resolvers to fill up the cache.
sl@0
   730
	2. Use a fifth one to bump off the least recently used entry.
sl@0
   731
    Run these two steps with the LRU entry in different positions in the
sl@0
   732
	queue, i.e. it is first, second, and last in the queue.
sl@0
   733
@SYMTestExpectedResults 1. The cache has 4 entries and is sorted in UID order.
sl@0
   734
	2. The oldest entry is gone from cache. The last one used is in cache.
sl@0
   735
@SYMCR CR1182
sl@0
   736
*/
sl@0
   737
void CCustomResolverCacheTest::TestCacheQueueFullPattern1L()
sl@0
   738
	{
sl@0
   739
	test.Next(_L(" @SYMTestCaseID:SYSLIB-ECOM-UT-4015 "));
sl@0
   740
	RArray<TUid> resolverlist(1);
sl@0
   741
	CleanupClosePushL(resolverlist);
sl@0
   742
sl@0
   743
	// NB: the following insertion order caused the problem found in
sl@0
   744
	// INC115472. Hence there will be a check in DoQueueFullTestL
sl@0
   745
	// that the cache queue is sorted correctly.
sl@0
   746
	//
sl@0
   747
	// Condition for inc115472 is first insert an UID of 0xA???????
sl@0
   748
	// in an empty queue. Next insert an UID 0x2???????. The overflow
sl@0
   749
	// error causes them to be placed in the wrong order.
sl@0
   750
	resolverlist.AppendL(KDummyResolverUid2);
sl@0
   751
	resolverlist.AppendL(KDummyResolverUid1);
sl@0
   752
	resolverlist.AppendL(KExampleResolverUid);
sl@0
   753
	resolverlist.AppendL(KMyResolverUid);
sl@0
   754
	resolverlist.AppendL(KDummyResolverUid3);
sl@0
   755
	DoQueueFullTestL(resolverlist);
sl@0
   756
	CleanupStack::PopAndDestroy(&resolverlist);
sl@0
   757
	}
sl@0
   758
sl@0
   759
// In pattern 2 the oldest entry in cache has the largest UID value.
sl@0
   760
void CCustomResolverCacheTest::TestCacheQueueFullPattern2L()
sl@0
   761
	{
sl@0
   762
	test.Next(_L(" @SYMTestCaseID:SYSLIB-ECOM-UT-4015 "));
sl@0
   763
	RArray<TUid> resolverlist(1);
sl@0
   764
	CleanupClosePushL(resolverlist);
sl@0
   765
	resolverlist.AppendL(KDummyResolverUid1);
sl@0
   766
	resolverlist.AppendL(KDummyResolverUid2);
sl@0
   767
	resolverlist.AppendL(KDummyResolverUid3);
sl@0
   768
	resolverlist.AppendL(KExampleResolverUid);
sl@0
   769
	resolverlist.AppendL(KMyResolverUid);
sl@0
   770
	DoQueueFullTestL(resolverlist);
sl@0
   771
	CleanupStack::PopAndDestroy(&resolverlist);
sl@0
   772
	}
sl@0
   773
sl@0
   774
// In pattern 3 the UID of the oldest entry has the second smallest value.
sl@0
   775
void CCustomResolverCacheTest::TestCacheQueueFullPattern3L()
sl@0
   776
	{
sl@0
   777
	test.Next(_L(" @SYMTestCaseID:SYSLIB-ECOM-UT-4015 "));
sl@0
   778
	RArray<TUid> resolverlist(1);
sl@0
   779
	CleanupClosePushL(resolverlist);
sl@0
   780
	resolverlist.AppendL(KDummyResolverUid3);
sl@0
   781
	resolverlist.AppendL(KDummyResolverUid1);
sl@0
   782
	resolverlist.AppendL(KDummyResolverUid2);
sl@0
   783
	resolverlist.AppendL(KExampleResolverUid);
sl@0
   784
	resolverlist.AppendL(KMyResolverUid);
sl@0
   785
	DoQueueFullTestL(resolverlist);
sl@0
   786
	CleanupStack::PopAndDestroy(&resolverlist);
sl@0
   787
	}
sl@0
   788
sl@0
   789
void CCustomResolverCacheTest::DoQueueFullTestL(const RArray<TUid>& aResolverList)
sl@0
   790
	{
sl@0
   791
	// ensure cache is empty.
sl@0
   792
	test(CacheSize() == 0);
sl@0
   793
sl@0
   794
	TInt maxQueueSize = ResolverCache()->iMaxCacheSize;
sl@0
   795
	test(aResolverList.Count() > maxQueueSize);
sl@0
   796
sl@0
   797
	// Put 4 resolvers in the cache. Pause in between so that the time tick
sl@0
   798
	// of each entry is different. CacheMissPerfTestL in t_resolverperf exercises
sl@0
   799
	// the code path that time ticks are the same.
sl@0
   800
	const TInt KInBetweenDelay = 100000;
sl@0
   801
	TInt i;
sl@0
   802
	for (i = 0; i < maxQueueSize; i++)
sl@0
   803
		{
sl@0
   804
		UseResolverCheckVersionL(aResolverList[i], 0);
sl@0
   805
		User::AfterHighRes(KInBetweenDelay);
sl@0
   806
		}
sl@0
   807
sl@0
   808
	// Check cache is full.
sl@0
   809
	test(CacheSize() == maxQueueSize);
sl@0
   810
	// Check cache is sorted properly
sl@0
   811
	test( QueueIsSorted() );
sl@0
   812
sl@0
   813
	// Add one more to cache.
sl@0
   814
	UseResolverCheckVersionL(aResolverList[maxQueueSize], 0);
sl@0
   815
sl@0
   816
	// Check that cache size is still full - not exceeding limit.
sl@0
   817
	test(CacheSize() == maxQueueSize);
sl@0
   818
sl@0
   819
	// Want to check LRU is gone and the last resolver is now in cache.
sl@0
   820
	test(! HasResolverUid(aResolverList[0]));
sl@0
   821
	test( QueueIsSorted() );
sl@0
   822
	// Because of the above 3 checks, this is really redundant.
sl@0
   823
	test(HasResolverUid(aResolverList[maxQueueSize]));
sl@0
   824
sl@0
   825
	// Because the timestamps of the cache entries are staggered
sl@0
   826
	// 100 ms apart, the mgr can only kick out 1 item at each timer expiry.
sl@0
   827
	// So let the cache mgr exercise this code path.
sl@0
   828
	TInt n = 0;
sl@0
   829
	while (CacheSize() > 0)
sl@0
   830
		{
sl@0
   831
		AsyncEvents.ClearAll();
sl@0
   832
		YieldToOtherCActive(KIndefiniteWait);  // 20 s
sl@0
   833
		// Check which async event has stopped the activescheduler
sl@0
   834
		test( AsyncEvents.IsSet(EAsyncEvent_CacheTimer) );
sl@0
   835
		n++;
sl@0
   836
		}
sl@0
   837
sl@0
   838
	// n should be five. aResolverList[0] is bumped due to queue full.
sl@0
   839
	// So the first timer expire will not find any expired entry.
sl@0
   840
	test.Printf(_L("Gone through %d loops to clear out cache.\n"), n);
sl@0
   841
sl@0
   842
	test(n > maxQueueSize);
sl@0
   843
	}
sl@0
   844
sl@0
   845
/**
sl@0
   846
@SYMTestCaseID		SYSLIB-ECOM-UT-4016
sl@0
   847
@SYMTestCaseDesc 	Verify resolver is unloaded after the specified timeout.
sl@0
   848
@SYMTestPriority 	Medium
sl@0
   849
@SYMTestActions  	1. Use a resolver to get it in cache.
sl@0
   850
	2. Measure how long the entry stays in cache before it is evicted.
sl@0
   851
@SYMTestExpectedResults The time should be >= default timeout but
sl@0
   852
	<= (timeout + 0.5s).
sl@0
   853
@SYMCR CR1182
sl@0
   854
*/
sl@0
   855
void CCustomResolverCacheTest::TestCacheTimerAccuracyL()
sl@0
   856
	{
sl@0
   857
	test.Next(_L(" @SYMTestCaseID:SYSLIB-ECOM-UT-4016 "));
sl@0
   858
	TTime start;
sl@0
   859
	start.UniversalTime();
sl@0
   860
sl@0
   861
	UseResolverCheckVersionL(KMyResolverUid, 0);
sl@0
   862
	test( HasResolverUid(KMyResolverUid) );
sl@0
   863
sl@0
   864
	// Wait for cache timeout
sl@0
   865
	AsyncEvents.ClearAll();
sl@0
   866
	YieldToOtherCActive(KIndefiniteWait);  // 20 s
sl@0
   867
	// Check which async event has stopped the activescheduler
sl@0
   868
	test( AsyncEvents.IsSet(EAsyncEvent_CacheTimer) );
sl@0
   869
sl@0
   870
	TInt microseconds = CalcElapsedMicroseconds(start);
sl@0
   871
	test.Printf(_L("Resolver is cached for %d microseconds\n"), microseconds);
sl@0
   872
sl@0
   873
	TInt tickperiod;
sl@0
   874
	User::LeaveIfError(HAL::Get(HALData::ESystemTickPeriod, tickperiod));
sl@0
   875
	TInt lowerLimit = KCustomResolverCacheTimeout - tickperiod;
sl@0
   876
	test(microseconds > lowerLimit);
sl@0
   877
sl@0
   878
	// The upper bound is tricky because there is no gaurantee on
sl@0
   879
	// CTimer accuracy.
sl@0
   880
	test(microseconds < (KCustomResolverCacheTimeout + KHalfSecond));
sl@0
   881
	}
sl@0
   882
sl@0
   883
/**
sl@0
   884
@SYMTestCaseID		SYSLIB-ECOM-UT-4017
sl@0
   885
@SYMTestCaseDesc 	Verify after a cache hit the time to live of the entry
sl@0
   886
	is extended.
sl@0
   887
@SYMTestPriority 	High
sl@0
   888
@SYMTestActions  	1. Record start time.
sl@0
   889
	2. Use a resolver to put it in cache.
sl@0
   890
	3. After 1 s use it again.
sl@0
   891
	4  Wait for its eviction.
sl@0
   892
	5. Check how long the entry stays in cache.
sl@0
   893
@SYMTestExpectedResults The time should be >= 5 seconds.
sl@0
   894
@SYMCR CR1182
sl@0
   895
*/
sl@0
   896
void CCustomResolverCacheTest::TestTimestampUpdateOnCacheHitL()
sl@0
   897
	{
sl@0
   898
	test.Next(_L(" @SYMTestCaseID:SYSLIB-ECOM-UT-4017 "));
sl@0
   899
	// Check that the resolver is not in cache.
sl@0
   900
	test(! HasResolverUid(KMyResolverUid) );
sl@0
   901
sl@0
   902
	TTime start;
sl@0
   903
	start.UniversalTime();
sl@0
   904
sl@0
   905
	UseResolverCheckVersionL(KMyResolverUid, 0);
sl@0
   906
	test( HasResolverUid(KMyResolverUid) );
sl@0
   907
sl@0
   908
	// Delay one second and use it again.
sl@0
   909
	User::AfterHighRes(KOneSecond);
sl@0
   910
	UseResolverCheckVersionL(KMyResolverUid, 0);
sl@0
   911
sl@0
   912
	// Wait for its eviction
sl@0
   913
	while ( HasResolverUid(KMyResolverUid) )
sl@0
   914
		{
sl@0
   915
		AsyncEvents.ClearAll();
sl@0
   916
		YieldToOtherCActive(KIndefiniteWait);  // 20 s
sl@0
   917
		// Check which async event has stopped the activescheduler
sl@0
   918
		test( AsyncEvents.IsSet(EAsyncEvent_CacheTimer) );
sl@0
   919
		}
sl@0
   920
sl@0
   921
	TInt microseconds = CalcElapsedMicroseconds(start);
sl@0
   922
	test.Printf(_L("With cache hit, resolver is cached for %d microseconds\n"), microseconds);
sl@0
   923
sl@0
   924
	TInt tickperiod;
sl@0
   925
	User::LeaveIfError(HAL::Get(HALData::ESystemTickPeriod, tickperiod));
sl@0
   926
	TInt lowerLimit = KCustomResolverCacheTimeout + KOneSecond - tickperiod;
sl@0
   927
	test(microseconds > lowerLimit);
sl@0
   928
	}
sl@0
   929
sl@0
   930
/**
sl@0
   931
@SYMTestCaseID		SYSLIB-ECOM-UT-4018
sl@0
   932
@SYMTestCaseDesc 	Verify at SWI start RW resolvers are evicted from cache.
sl@0
   933
	Verify during SWI ROM resolvers are cached but RW resolvers are not.
sl@0
   934
@SYMTestPriority 	High
sl@0
   935
@SYMTestActions  	1. Add a RW resolver to cache.
sl@0
   936
	2. Send a SWI start signal.
sl@0
   937
	3. Use a RW resolver.
sl@0
   938
	4  Use a ROM resolver.
sl@0
   939
@SYMTestExpectedResults 1. The RW resolver is in cache.
sl@0
   940
	2. The cache entry is evicted.
sl@0
   941
	3. The RW resolver is not cached.
sl@0
   942
	4. The ROM resolver is cached.
sl@0
   943
@SYMCR CR1182
sl@0
   944
*/
sl@0
   945
void CCustomResolverCacheTest::TestSWIDisableRwResolverCachingL()
sl@0
   946
	{
sl@0
   947
	test.Next(_L(" @SYMTestCaseID:SYSLIB-ECOM-UT-4018 "));
sl@0
   948
	// The TPropertyManager protocol to setup to use a system category P&S.
sl@0
   949
	TInt r = PropertyManager::DeleteProperty(KUidSystemCategory,
sl@0
   950
		KSAUidSoftwareInstallKeyValue);
sl@0
   951
	test(r == KErrNone);
sl@0
   952
	r = PropertyManager::DefineProperty(KUidSystemCategory,
sl@0
   953
		KSAUidSoftwareInstallKeyValue, RProperty::EInt);
sl@0
   954
	test(r == KErrNone);
sl@0
   955
sl@0
   956
	// Use a resolver in C: drive.
sl@0
   957
	UseResolverCheckVersionL(KDummyResolverUid1, 0);
sl@0
   958
	// Check that it is added to cache.
sl@0
   959
	test( HasResolverUid(KDummyResolverUid1) );
sl@0
   960
sl@0
   961
	// Need a ROM entry in cache to get Bullseye to check off
sl@0
   962
	// a conditional. It is not an active ingredient of this test.
sl@0
   963
	UseResolverCheckVersionL(KMyResolverUid, 0);
sl@0
   964
sl@0
   965
	// SWI start
sl@0
   966
	r = PropertyManager::SetProperty(KUidSystemCategory,
sl@0
   967
		KSAUidSoftwareInstallKeyValue,ESASwisInstall);
sl@0
   968
	test(r == KErrNone);
sl@0
   969
sl@0
   970
	// Let CSwiChangeNotifier receive the notification
sl@0
   971
	AsyncEvents.ClearAll();
sl@0
   972
	YieldToOtherCActive(KIndefiniteWait);  // 20 s
sl@0
   973
	// Check we receive the correct event
sl@0
   974
	test( AsyncEvents.IsSet(EAsyncEvent_SwiStart) );
sl@0
   975
sl@0
   976
	// Check the RW resolver is evicted
sl@0
   977
	test(! HasResolverUid(KDummyResolverUid1) );
sl@0
   978
sl@0
   979
	// during SWI RW resolver will not be cached.
sl@0
   980
	UseResolverCheckVersionL(KDummyResolverUid1, 0);
sl@0
   981
	test(! HasResolverUid(KDummyResolverUid1) );
sl@0
   982
sl@0
   983
	// But ROM resolvers are still cached.
sl@0
   984
	UseResolverCheckVersionL(KDummyResolverUid2, 0);
sl@0
   985
	test( HasResolverUid(KDummyResolverUid2) );
sl@0
   986
sl@0
   987
	// And the ROM resolvers are evicted after 4 s as usual.
sl@0
   988
	// Note that we have KMyResolverUid and KDummyResolverUid2 in cache
sl@0
   989
	TBool b = WaitForEvict(KDummyResolverUid2);
sl@0
   990
	if (! b)
sl@0
   991
		{ // got KMyResolverUid only, need a second wait.
sl@0
   992
		b = WaitForEvict(KDummyResolverUid2);
sl@0
   993
		}
sl@0
   994
	test(b);
sl@0
   995
sl@0
   996
	// Just for completeness. Do not really need this
sl@0
   997
	// for Bullseye Coverage.
sl@0
   998
	r = PropertyManager::SetProperty(KUidSystemCategory,
sl@0
   999
		KSAUidSoftwareInstallKeyValue,ESASwisNone);
sl@0
  1000
	test(r == KErrNone);
sl@0
  1001
	// Let CSwiChangeNotifier receive the notification
sl@0
  1002
	AsyncEvents.ClearAll();
sl@0
  1003
	YieldToOtherCActive(KIndefiniteWait);  // 20 s
sl@0
  1004
	// Check we receive the correct event
sl@0
  1005
	test( AsyncEvents.IsSet(EAsyncEvent_SwiEnd) );
sl@0
  1006
	}
sl@0
  1007
sl@0
  1008
/**
sl@0
  1009
@SYMTestCaseID		SYSLIB-ECOM-UT-4019
sl@0
  1010
@SYMTestCaseDesc 	Verify at BUR start RW resolvers are evicted from cache.
sl@0
  1011
	Verify during BUR ROM resolvers are cached but RW resolvers are not.
sl@0
  1012
@SYMTestPriority 	High
sl@0
  1013
@SYMTestActions  	1. Add a RW resolver to cache.
sl@0
  1014
	2. Send a BUR start signal.
sl@0
  1015
	3. Use a RW resolver.
sl@0
  1016
	4  Use a ROM resolver.
sl@0
  1017
@SYMTestExpectedResults 1. The RW resolver is in cache.
sl@0
  1018
	2. The cache entry is evicted.
sl@0
  1019
	3. The RW resolver is not cached.
sl@0
  1020
	4. The ROM resolver is cached.
sl@0
  1021
@SYMCR CR1182
sl@0
  1022
*/
sl@0
  1023
void CCustomResolverCacheTest::TestBurDisableRwResolverCachingL()
sl@0
  1024
	{
sl@0
  1025
	test.Next(_L(" @SYMTestCaseID:SYSLIB-ECOM-UT-4019 "));
sl@0
  1026
	// setup backup session wrapper
sl@0
  1027
	CBaBackupSessionWrapper* ba = CBaBackupSessionWrapper::NewL();
sl@0
  1028
	CleanupStack::PushL(ba);
sl@0
  1029
	TBackupOperationAttributes attribs;
sl@0
  1030
	attribs.iFileFlag=MBackupObserver::EReleaseLockNoAccess;
sl@0
  1031
	attribs.iOperation=MBackupOperationObserver::EStart;
sl@0
  1032
sl@0
  1033
	// The backup notifier of ECOM is not registered with
sl@0
  1034
	// BA server at construction time. It delays 15 s.
sl@0
  1035
	// So we have to wait this long.
sl@0
  1036
	WaitAsyncL(16);
sl@0
  1037
sl@0
  1038
	// Use a resolver in C: drive.
sl@0
  1039
	UseResolverCheckVersionL(KDummyResolverUid1, 0);
sl@0
  1040
	// Check that it is added to cache.
sl@0
  1041
	test( HasResolverUid(KDummyResolverUid1) );
sl@0
  1042
sl@0
  1043
	// Need a ROM entry in cache to get Bullseye to check off
sl@0
  1044
	// a conditional. It is not an active ingredient of this test.
sl@0
  1045
	UseResolverCheckVersionL(KMyResolverUid, 0);
sl@0
  1046
sl@0
  1047
	// backup start
sl@0
  1048
	ba->NotifyBackupOperationL(attribs);
sl@0
  1049
	// Let backup notifier receive the backup start
sl@0
  1050
	AsyncEvents.ClearAll();
sl@0
  1051
	YieldToOtherCActive(KIndefiniteWait);  // 20 s
sl@0
  1052
	// Check we receive the backup start notification
sl@0
  1053
	test( AsyncEvents.IsSet(EAsyncEvent_BurStart) );
sl@0
  1054
sl@0
  1055
	// Check the RW resolver is evicted
sl@0
  1056
	test(! HasResolverUid(KDummyResolverUid1) );
sl@0
  1057
sl@0
  1058
	// during BUR RW resolver will not be cached.
sl@0
  1059
	UseResolverCheckVersionL(KDummyResolverUid1, 0);
sl@0
  1060
	test(! HasResolverUid(KDummyResolverUid1) );
sl@0
  1061
sl@0
  1062
	// But ROM resolvers are still cached.
sl@0
  1063
	UseResolverCheckVersionL(KDummyResolverUid2, 0);
sl@0
  1064
	test( HasResolverUid(KDummyResolverUid2) );
sl@0
  1065
sl@0
  1066
	// And the ROM resolvers are evicted after 4 s as usual.
sl@0
  1067
	// Note that we have KMyResolverUid and KDummyResolverUid2 in cache
sl@0
  1068
	TBool b = WaitForEvict(KDummyResolverUid2);
sl@0
  1069
	if (! b)
sl@0
  1070
		{ // got KMyResolverUid only, need a second wait.
sl@0
  1071
		b = WaitForEvict(KDummyResolverUid2);
sl@0
  1072
		}
sl@0
  1073
	test(b);
sl@0
  1074
sl@0
  1075
	// Do this for Bullseye Coverage of production code.
sl@0
  1076
	// NB: attribs.iFileFlag stays the same
sl@0
  1077
	attribs.iOperation=MBackupOperationObserver::EEnd;
sl@0
  1078
	ba->NotifyBackupOperationL(attribs);
sl@0
  1079
	// Let backup notifier receive it
sl@0
  1080
	AsyncEvents.ClearAll();
sl@0
  1081
	YieldToOtherCActive(KIndefiniteWait);  // 20 s
sl@0
  1082
	// Check we receive the backup notification
sl@0
  1083
	test( AsyncEvents.IsSet(EAsyncEvent_BurEnd) );
sl@0
  1084
sl@0
  1085
	CleanupStack::PopAndDestroy(ba);
sl@0
  1086
sl@0
  1087
	// to earn another Bullseye check mark.
sl@0
  1088
	CEComServer::NotifyEvents(iEComServer, ECallBackId_None, NULL);
sl@0
  1089
	}
sl@0
  1090
sl@0
  1091
/**
sl@0
  1092
@SYMTestCaseID		SYSLIB-ECOM-UT-4021
sl@0
  1093
@SYMTestCaseDesc 	Verify the cache timer and data to keep track of
sl@0
  1094
	cache entry time to live are immune to device clock change.
sl@0
  1095
@SYMTestPriority 	High
sl@0
  1096
@SYMTestActions  	1. Put two resolvers in cache at 20 ticks apart.
sl@0
  1097
	2. Spring clock forward by 5 s (1 + default cache timeout).
sl@0
  1098
	3. Measure how long the first resolver is cached.
sl@0
  1099
	4. Measure how long the second resolver is cached.
sl@0
  1100
	Repeat steps 1 to 4 in setting clock backward.
sl@0
  1101
@SYMTestExpectedResults Both resolvers are cached for ~ 4 s
sl@0
  1102
	(default cache timeout value).
sl@0
  1103
@SYMCR CR1182
sl@0
  1104
*/
sl@0
  1105
void CCustomResolverCacheTest::TestClockChangeHasNoEffectOnCacheTimeoutL()
sl@0
  1106
	{
sl@0
  1107
	test.Next(_L(" @SYMTestCaseID:SYSLIB-ECOM-UT-4021 "));
sl@0
  1108
	TInt tp;
sl@0
  1109
	User::LeaveIfError(HAL::Get(HALData::ESystemTickPeriod, tp));
sl@0
  1110
	TInt delayInBetween = tp * 20;
sl@0
  1111
   	test.Printf(_L("ESystemTickPeriod: %d; delayInBetween: %d;\n"), tp, delayInBetween);
sl@0
  1112
sl@0
  1113
sl@0
  1114
	TTime beforeDelay;
sl@0
  1115
	beforeDelay.UniversalTime();
sl@0
  1116
sl@0
  1117
	UseResolverCheckVersionL(KDummyResolverUid1, 0);
sl@0
  1118
	test( HasResolverUid(KDummyResolverUid1) );
sl@0
  1119
sl@0
  1120
	User::AfterHighRes(delayInBetween);
sl@0
  1121
	TInt elapsedBeforeTimeChange = CalcElapsedMicroseconds(beforeDelay);
sl@0
  1122
sl@0
  1123
	UseResolverCheckVersionL(KDummyResolverUid2, 0);
sl@0
  1124
	test( HasResolverUid(KDummyResolverUid2) );
sl@0
  1125
sl@0
  1126
	TTimeIntervalMicroSeconds AmountOfChange(
sl@0
  1127
		MAKE_TINT64(0, KCustomResolverCacheTimeout+KOneSecond) );
sl@0
  1128
sl@0
  1129
	// Now set device clock forward
sl@0
  1130
	TTime t;
sl@0
  1131
	t.UniversalTime();
sl@0
  1132
	t += AmountOfChange;
sl@0
  1133
	User::LeaveIfError( User::SetUTCTime(t) );
sl@0
  1134
sl@0
  1135
	t.UniversalTime(); // need this, apparently SetUTCTime does rounding
sl@0
  1136
sl@0
  1137
	// Wait for cache timeout
sl@0
  1138
	AsyncEvents.ClearAll();
sl@0
  1139
	YieldToOtherCActive(KIndefiniteWait);  // 20 s
sl@0
  1140
	// Check which async event has stopped the activescheduler
sl@0
  1141
	test( AsyncEvents.IsSet(EAsyncEvent_CacheTimer) );
sl@0
  1142
sl@0
  1143
	TInt microseconds = CalcElapsedMicroseconds(t);
sl@0
  1144
	// compensate for the 20 ticks delay
sl@0
  1145
	microseconds += elapsedBeforeTimeChange;
sl@0
  1146
	test.Printf(_L("Clock forward: 1st resolver is cached for %d microseconds\n"), microseconds);
sl@0
  1147
	test(! HasResolverUid(KDummyResolverUid1) );
sl@0
  1148
	// Check duration is between 4 to 4.5 s.
sl@0
  1149
	TInt lowerLimit = KCustomResolverCacheTimeout - (tp*3);
sl@0
  1150
	TInt upperLimit = (KCustomResolverCacheTimeout + KHalfSecond);
sl@0
  1151
   	test.Printf(_L("lowerLimit: %d; upperLimit: %d\n"), lowerLimit, upperLimit);	
sl@0
  1152
	test(microseconds > lowerLimit);
sl@0
  1153
sl@0
  1154
	// upper limit is tricky because there is no guarantee on CTimer
sl@0
  1155
	// accuracy.
sl@0
  1156
	test(microseconds <= upperLimit);
sl@0
  1157
sl@0
  1158
	// Wait for eviction of the second resolver
sl@0
  1159
	AsyncEvents.ClearAll();
sl@0
  1160
	YieldToOtherCActive(KIndefiniteWait);  // 20 s
sl@0
  1161
	// Check which async event has stopped the activescheduler
sl@0
  1162
	test( AsyncEvents.IsSet(EAsyncEvent_CacheTimer) );
sl@0
  1163
sl@0
  1164
	microseconds = CalcElapsedMicroseconds(t);
sl@0
  1165
	test.Printf(_L("Clock forward: 2nd resolver is cached for %d microseconds\n"), microseconds);
sl@0
  1166
	test(! HasResolverUid(KDummyResolverUid2) );
sl@0
  1167
	// Check duration is between 4 to 4.5 s.
sl@0
  1168
	test(microseconds > lowerLimit);
sl@0
  1169
	test(microseconds <= upperLimit);
sl@0
  1170
sl@0
  1171
	//
sl@0
  1172
	// Repeat for setting clock backward.
sl@0
  1173
	//
sl@0
  1174
	beforeDelay.UniversalTime();
sl@0
  1175
sl@0
  1176
	UseResolverCheckVersionL(KDummyResolverUid1, 0);
sl@0
  1177
	test( HasResolverUid(KDummyResolverUid1) );
sl@0
  1178
sl@0
  1179
	User::AfterHighRes(delayInBetween);
sl@0
  1180
	elapsedBeforeTimeChange = CalcElapsedMicroseconds(beforeDelay);
sl@0
  1181
sl@0
  1182
	UseResolverCheckVersionL(KDummyResolverUid2, 0);
sl@0
  1183
	test( HasResolverUid(KDummyResolverUid2) );
sl@0
  1184
sl@0
  1185
	// Now set device clock backward
sl@0
  1186
	t.UniversalTime();
sl@0
  1187
	t -= AmountOfChange;
sl@0
  1188
	User::LeaveIfError( User::SetUTCTime(t) );
sl@0
  1189
	t.UniversalTime();
sl@0
  1190
sl@0
  1191
	// Wait for cache timeout
sl@0
  1192
	AsyncEvents.ClearAll();
sl@0
  1193
	YieldToOtherCActive(KIndefiniteWait);  // 20 s
sl@0
  1194
	// Check which async event has stopped the activescheduler
sl@0
  1195
	test( AsyncEvents.IsSet(EAsyncEvent_CacheTimer) );
sl@0
  1196
sl@0
  1197
	microseconds = CalcElapsedMicroseconds(t);
sl@0
  1198
	// compensate for the 20 ticks delay
sl@0
  1199
	microseconds += elapsedBeforeTimeChange;
sl@0
  1200
	test.Printf(_L("Clock backward: 1st resolver is cached for %d microseconds\n"), microseconds);
sl@0
  1201
	test(! HasResolverUid(KDummyResolverUid1) );
sl@0
  1202
	// Check duration is between 4 to 4.5 s.
sl@0
  1203
	test(microseconds > lowerLimit);
sl@0
  1204
	test(microseconds <= (KCustomResolverCacheTimeout + KHalfSecond));
sl@0
  1205
sl@0
  1206
	// Wait for eviction of the second resolver
sl@0
  1207
	AsyncEvents.ClearAll();
sl@0
  1208
	YieldToOtherCActive(KIndefiniteWait);  // 20 s
sl@0
  1209
	// Check which async event has stopped the activescheduler
sl@0
  1210
	test( AsyncEvents.IsSet(EAsyncEvent_CacheTimer) );
sl@0
  1211
sl@0
  1212
	microseconds = CalcElapsedMicroseconds(t);
sl@0
  1213
	test.Printf(_L("Clock backward: 2nd resolver is cached for %d microseconds\n"), microseconds);
sl@0
  1214
	test(! HasResolverUid(KDummyResolverUid2) );
sl@0
  1215
	// Check duration is between 4 to 4.5 s.
sl@0
  1216
	test(microseconds > lowerLimit);
sl@0
  1217
	test(microseconds <= (KCustomResolverCacheTimeout + KHalfSecond));
sl@0
  1218
	}
sl@0
  1219
sl@0
  1220
/**
sl@0
  1221
@SYMTestCaseID		SYSLIB-ECOM-UT-4022
sl@0
  1222
@SYMTestCaseDesc 	Verify caching runs normally if queue size is zero.
sl@0
  1223
@SYMTestPriority 	Medium
sl@0
  1224
@SYMTestActions  	1. Set max. cache size to 0.
sl@0
  1225
	2. Use a custom resolver.
sl@0
  1226
@SYMTestExpectedResults No leave on the list request and cache is empty.
sl@0
  1227
@SYMCR CR1182
sl@0
  1228
*/
sl@0
  1229
void CCustomResolverCacheTest::TestCacheSizeZeroL()
sl@0
  1230
	{
sl@0
  1231
	test.Next(_L(" @SYMTestCaseID:SYSLIB-ECOM-UT-4022 "));
sl@0
  1232
	ResolverCache()->iMaxCacheSize = 0;
sl@0
  1233
	UseResolverCheckVersionL(KMyResolverUid, 0);
sl@0
  1234
	test(! HasResolverUid(KMyResolverUid) );
sl@0
  1235
	}
sl@0
  1236
sl@0
  1237
/**
sl@0
  1238
@SYMTestCaseID		SYSLIB-ECOM-UT-4023
sl@0
  1239
@SYMTestCaseDesc 	Verify caching runs normally if cache timeout is zero.
sl@0
  1240
@SYMTestPriority 	Medium
sl@0
  1241
@SYMTestActions  	1. Set cache time-to-live to zero.
sl@0
  1242
	2. Use a custom resolver.
sl@0
  1243
@SYMTestExpectedResults No leave on the list request and cache is empty.
sl@0
  1244
@SYMCR CR1182
sl@0
  1245
*/
sl@0
  1246
void CCustomResolverCacheTest::TestCacheTimeoutZeroL()
sl@0
  1247
	{
sl@0
  1248
	test.Next(_L(" @SYMTestCaseID:SYSLIB-ECOM-UT-4023 "));
sl@0
  1249
	TInt systemTickPeriod = ResolverCache()->iSystemTickPeriod;
sl@0
  1250
	// mimic CCustomResolverCache calculate the time-to-live value.
sl@0
  1251
	const TInt KZeroTimeout = 0;
sl@0
  1252
	ResolverCache()->iEntryTimeToLive = (KZeroTimeout +
sl@0
  1253
		systemTickPeriod - 1) / systemTickPeriod;
sl@0
  1254
sl@0
  1255
	// run the test
sl@0
  1256
	UseResolverCheckVersionL(KMyResolverUid, 0);
sl@0
  1257
	test(! HasResolverUid(KMyResolverUid) );
sl@0
  1258
	}
sl@0
  1259
sl@0
  1260
/**
sl@0
  1261
@SYMTestCaseID		SYSLIB-ECOM-UT-4024
sl@0
  1262
@SYMTestCaseDesc 	CCustomResolverCache uses system tick to track entry
sl@0
  1263
	time-to-live. The counter wraps around every 777 days. This test
sl@0
  1264
	checks when wrap around occurs cached entry is purged correctly.
sl@0
  1265
@SYMTestPriority 	Medium
sl@0
  1266
@SYMTestActions  	1. Use a resolver to get it in cache.
sl@0
  1267
	2. Set its time-of-use to 0xFFFFFFFF.
sl@0
  1268
	3. Wait for cache timer expire.
sl@0
  1269
@SYMTestExpectedResults The entry is evicted on first timer expiry and
sl@0
  1270
	the entry stays in cache for ~ 4s.
sl@0
  1271
@SYMCR CR1182
sl@0
  1272
*/
sl@0
  1273
void CCustomResolverCacheTest::TestCounterWrapAroundL()
sl@0
  1274
	{
sl@0
  1275
	test.Next(_L(" @SYMTestCaseID:SYSLIB-ECOM-UT-4024 "));
sl@0
  1276
	TTime start;
sl@0
  1277
	start.UniversalTime();
sl@0
  1278
sl@0
  1279
	UseResolverCheckVersionL(KMyResolverUid, 0);
sl@0
  1280
	RResolverCacheEntry& entry = ResolverCache()->iResolvers[0];
sl@0
  1281
	test( entry.iResolverUid == KMyResolverUid );
sl@0
  1282
sl@0
  1283
	// Set time-of-use to edge of wrap around
sl@0
  1284
	entry.iLastUse.iTicks = KMaxTUint;
sl@0
  1285
sl@0
  1286
	// Wait for cache timeout
sl@0
  1287
	AsyncEvents.ClearAll();
sl@0
  1288
	YieldToOtherCActive(KIndefiniteWait);  // 20 s
sl@0
  1289
	// Check which async event has stopped the activescheduler
sl@0
  1290
	test( AsyncEvents.IsSet(EAsyncEvent_CacheTimer) );
sl@0
  1291
sl@0
  1292
	TInt microseconds = CalcElapsedMicroseconds(start);
sl@0
  1293
	test.Printf(_L("Wrap around: resolver is cached for %d microseconds\n"), microseconds);
sl@0
  1294
sl@0
  1295
	TInt tickperiod;
sl@0
  1296
	User::LeaveIfError(HAL::Get(HALData::ESystemTickPeriod, tickperiod));
sl@0
  1297
	TInt lowerLimit = KCustomResolverCacheTimeout - tickperiod;
sl@0
  1298
	test(microseconds > lowerLimit);
sl@0
  1299
	// Upper limit must have a wide margin because there is no gaurantee
sl@0
  1300
	// on CTimer accuracy.
sl@0
  1301
	test(microseconds < (KCustomResolverCacheTimeout + KHalfSecond));
sl@0
  1302
	}
sl@0
  1303
sl@0
  1304
/**
sl@0
  1305
@SYMTestCaseID		SYSLIB-ECOM-UT-4025
sl@0
  1306
@SYMTestCaseDesc 	Verify if cannot find the NewL method in the proxy
sl@0
  1307
	instantiation table, ECOM returns KEComErrNoResolver
sl@0
  1308
@SYMTestPriority 	Medium
sl@0
  1309
@SYMTestActions  	Use a resolver whose proxy table has the wrong Impl. Uid.
sl@0
  1310
@SYMTestExpectedResults Get KEComErrNoResolver error
sl@0
  1311
@SYMCR CR1182
sl@0
  1312
*/
sl@0
  1313
void CCustomResolverCacheTest::TestResolverWithBadProxyTable()
sl@0
  1314
	{
sl@0
  1315
	test.Next(_L(" @SYMTestCaseID:SYSLIB-ECOM-UT-4025 "));
sl@0
  1316
	const TUid KBadResolverUid = {0x10009DDF};
sl@0
  1317
	TRAPD(err, UseResolverCheckVersionL(KBadResolverUid, 0) );
sl@0
  1318
	test(err == KEComErrNoResolver);
sl@0
  1319
	}
sl@0
  1320
sl@0
  1321
/** wrapper to run all the Basic and OOM tests */
sl@0
  1322
void CCustomResolverCacheTest::RunAllTestsL()
sl@0
  1323
	{
sl@0
  1324
	CopyPluginsL();
sl@0
  1325
sl@0
  1326
	test.Next(_L("Basic TestCounterWrapAroundL"));
sl@0
  1327
	DoBasicTestL(&CCustomResolverCacheTest::TestCounterWrapAroundL);
sl@0
  1328
sl@0
  1329
	test.Next(_L("Basic TestUpgradingCachedResolverL"));
sl@0
  1330
	DoBasicTestL(&CCustomResolverCacheTest::TestUpgradingCachedResolverL);
sl@0
  1331
sl@0
  1332
	test.Next(_L("Basic TestCacheQueueFullPattern1L"));
sl@0
  1333
	DoBasicTestL(&CCustomResolverCacheTest::TestCacheQueueFullPattern1L);
sl@0
  1334
sl@0
  1335
	test.Next(_L("Basic TestCacheQueueFullPattern2L"));
sl@0
  1336
	DoBasicTestL(&CCustomResolverCacheTest::TestCacheQueueFullPattern2L);
sl@0
  1337
sl@0
  1338
	test.Next(_L("Basic TestCacheQueueFullPattern3L"));
sl@0
  1339
	DoBasicTestL(&CCustomResolverCacheTest::TestCacheQueueFullPattern3L);
sl@0
  1340
sl@0
  1341
	test.Next(_L("Basic TestCacheTimerAccuracyL"));
sl@0
  1342
	DoBasicTestL(&CCustomResolverCacheTest::TestCacheTimerAccuracyL);
sl@0
  1343
sl@0
  1344
	test.Next(_L("Basic TestTimestampUpdateOnCacheHitL"));
sl@0
  1345
	DoBasicTestL(&CCustomResolverCacheTest::TestTimestampUpdateOnCacheHitL);
sl@0
  1346
sl@0
  1347
	test.Next(_L("Basic TestSWIDisableRwResolverCachingL"));
sl@0
  1348
	DoBasicTestL(&CCustomResolverCacheTest::TestSWIDisableRwResolverCachingL);
sl@0
  1349
sl@0
  1350
	test.Next(_L("Basic TestBurDisableRwResolverCachingL"));
sl@0
  1351
	DoBasicTestL(&CCustomResolverCacheTest::TestBurDisableRwResolverCachingL);
sl@0
  1352
sl@0
  1353
	test.Next(_L("Basic TestClockChangeHasNoEffectOnCacheTimeoutL"));
sl@0
  1354
	DoBasicTestL(&CCustomResolverCacheTest::TestClockChangeHasNoEffectOnCacheTimeoutL);
sl@0
  1355
sl@0
  1356
	test.Next(_L("Basic TestCacheSizeZeroL"));
sl@0
  1357
	DoBasicTestL(&CCustomResolverCacheTest::TestCacheSizeZeroL);
sl@0
  1358
sl@0
  1359
	test.Next(_L("Basic TestCacheTimeoutZeroL"));
sl@0
  1360
	DoBasicTestL(&CCustomResolverCacheTest::TestCacheTimeoutZeroL);
sl@0
  1361
sl@0
  1362
	test.Next(_L("Basic TestResolverWithBadProxyTable"));
sl@0
  1363
	DoBasicTestL(&CCustomResolverCacheTest::TestResolverWithBadProxyTable);
sl@0
  1364
sl@0
  1365
	// Only run OOM on tests which do not involve rescan dir.
sl@0
  1366
	test.Next(_L("OOM TestCacheQueueFullPattern3L"));
sl@0
  1367
	DoOOMTestL(&CCustomResolverCacheTest::TestCacheQueueFullPattern3L);
sl@0
  1368
sl@0
  1369
	test.Next(_L("OOM TestCacheTimerAccuracyL"));
sl@0
  1370
	DoOOMTestL(&CCustomResolverCacheTest::TestCacheTimerAccuracyL);
sl@0
  1371
sl@0
  1372
	test.Next(_L("OOM TestTimestampUpdateOnCacheHitL"));
sl@0
  1373
	DoOOMTestL(&CCustomResolverCacheTest::TestTimestampUpdateOnCacheHitL);
sl@0
  1374
sl@0
  1375
	// Do all tests affected by Lazy DLL unload last.
sl@0
  1376
	test.Next(_L("Basic TestDeletingCachedResolverL"));
sl@0
  1377
	DoBasicTestL(&CCustomResolverCacheTest::TestDeletingCachedResolverL);
sl@0
  1378
	}
sl@0
  1379
sl@0
  1380
static TInt KillEComServer()
sl@0
  1381
	{
sl@0
  1382
	//Need to ensure that the EComServer process is killed before even starting this test by using
sl@0
  1383
   	//the EComTestUtils library
sl@0
  1384
   	_LIT(KEComServerProcessName,"ecomserver");
sl@0
  1385
   	TRAPD(err, EComTestUtils::KillProcessL(KEComServerProcessName));
sl@0
  1386
   	return err;
sl@0
  1387
	}
sl@0
  1388
sl@0
  1389
GLDEF_C TInt E32Main()
sl@0
  1390
	{
sl@0
  1391
	__UHEAP_MARK;
sl@0
  1392
sl@0
  1393
	beginningOfTest.UniversalTime();
sl@0
  1394
sl@0
  1395
	test.Printf(_L("\n"));
sl@0
  1396
	test.Title();
sl@0
  1397
	test.Start( _L("Custom Resolver Cache") );
sl@0
  1398
sl@0
  1399
sl@0
  1400
	TheTrapCleanup = CTrapCleanup::New();
sl@0
  1401
	test(TheTrapCleanup != NULL);
sl@0
  1402
sl@0
  1403
	// Connect the file server instance
sl@0
  1404
	test(KErrNone == TheFs.Connect());
sl@0
  1405
sl@0
  1406
	TheActiveScheduler = new CDerivedActiveScheduler;
sl@0
  1407
	test(TheActiveScheduler != NULL);
sl@0
  1408
	CActiveScheduler::Install(TheActiveScheduler);
sl@0
  1409
sl@0
  1410
	TInt err = ::KillEComServer();
sl@0
  1411
	test(err == KErrNotFound || err == KErrNone);
sl@0
  1412
sl@0
  1413
	// Call the main tests
sl@0
  1414
	TRAP(err, CCustomResolverCacheTest::RunAllTestsL());
sl@0
  1415
	if (err != KErrNone) test.Printf(_L("RunAllTestsL() error, %d\n"), err);
sl@0
  1416
	test(err == KErrNone);
sl@0
  1417
sl@0
  1418
	// remove the RAMOnly files
sl@0
  1419
	TRAP_IGNORE( WaitForLazyUnloadingL() );
sl@0
  1420
	DeleteTestPlugin();
sl@0
  1421
	User::After(KOneSecond * 3);
sl@0
  1422
sl@0
  1423
	delete TheActiveScheduler;
sl@0
  1424
	delete TheTrapCleanup;
sl@0
  1425
sl@0
  1426
	TheFs.Close();
sl@0
  1427
sl@0
  1428
	test.End();
sl@0
  1429
	test.Close();
sl@0
  1430
sl@0
  1431
	__UHEAP_MARKEND;
sl@0
  1432
	return (KErrNone);
sl@0
  1433
	}