os/persistentdata/persistentstorage/centralrepository/test/T_CenRepTrans.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) 2005-2009 Nokia Corporation and/or its subsidiary(-ies).
sl@0
     2
// All rights reserved.
sl@0
     3
// This component and the accompanying materials are made available
sl@0
     4
// under the terms of "Eclipse Public License v1.0"
sl@0
     5
// which accompanies this distribution, and is available
sl@0
     6
// at the URL "http://www.eclipse.org/legal/epl-v10.html".
sl@0
     7
//
sl@0
     8
// Initial Contributors:
sl@0
     9
// Nokia Corporation - initial contribution.
sl@0
    10
//
sl@0
    11
// Contributors:
sl@0
    12
//
sl@0
    13
// Description:
sl@0
    14
//
sl@0
    15
sl@0
    16
#include "t_cenrep_helper.h"
sl@0
    17
#include <e32test.h>
sl@0
    18
#include <f32file.h>
sl@0
    19
#include "srvrepos_noc.h"
sl@0
    20
#include "srvres.h"
sl@0
    21
#include "cachemgr.h"
sl@0
    22
#include <bautils.h>
sl@0
    23
#include "setting.h"
sl@0
    24
sl@0
    25
//Forward declarations
sl@0
    26
class CTestObserver;
sl@0
    27
sl@0
    28
///////////////////////////////////////////////////////////////////////////////////////
sl@0
    29
// Globals
sl@0
    30
sl@0
    31
static RTest TheTest(_L("T_CenRepTrans"));
sl@0
    32
const TUid KTestRepositoryId = {0x00000010};
sl@0
    33
//CentralRepository server UID
sl@0
    34
static const TUid KCenRepServerUID = {0x10202BE9};
sl@0
    35
sl@0
    36
static CTestObserver* TheObserver = NULL;
sl@0
    37
static CServerRepository* TheServerRepository = NULL;
sl@0
    38
sl@0
    39
static const TUint32 KIntSettingID = 843;
sl@0
    40
static const TInt KIntSettingVal = 25;
sl@0
    41
sl@0
    42
static const TUint32 KRealSettingID = 844;
sl@0
    43
static const TReal KRealSettingVal = 8.76;
sl@0
    44
static const TReal KRealSettingVal2 = 19.234;
sl@0
    45
sl@0
    46
static const TUint32 KStringSettingID = 845;
sl@0
    47
_LIT8(KStringSettingVal, "a test string");
sl@0
    48
_LIT8(KStringSettingVal2, "another string");
sl@0
    49
_LIT8(KEmptyString, "");
sl@0
    50
sl@0
    51
_LIT(KTestFile1,	"C:\\PRIVATE\\10202BE9\\PERSISTS\\00000010.TXT");
sl@0
    52
_LIT(KCreTestFile1,	"C:\\PRIVATE\\10202BE9\\PERSISTS\\00000010.CRE");
sl@0
    53
_LIT(KTmpTestFile1,	"C:\\PRIVATE\\10202BE9\\PERSISTS\\00000010.TMP");
sl@0
    54
sl@0
    55
sl@0
    56
///////////////////////////////////////////////////////////////////////////////////////
sl@0
    57
//
sl@0
    58
sl@0
    59
//Delete file function.
sl@0
    60
//Used by the test application to cleanup the C drive at the end of the test.
sl@0
    61
static void DeleteFile(const TDesC& aFullName)
sl@0
    62
	{
sl@0
    63
	RFs fsSession;
sl@0
    64
	TInt err = fsSession.Connect();
sl@0
    65
	if(err == KErrNone)
sl@0
    66
		{
sl@0
    67
		TEntry entry;
sl@0
    68
		if(fsSession.Entry(aFullName, entry) == KErrNone)
sl@0
    69
			{
sl@0
    70
			RDebug::Print(_L("Deleting \"%S\" file.\n"), &aFullName);
sl@0
    71
			err = fsSession.SetAtt(aFullName, 0, KEntryAttReadOnly);
sl@0
    72
			if(err != KErrNone)
sl@0
    73
				{
sl@0
    74
				RDebug::Print(_L("Error %d changing \"%S\" file attributes.\n"), err, &aFullName);
sl@0
    75
				}
sl@0
    76
			err = fsSession.Delete(aFullName);
sl@0
    77
			if(err != KErrNone)
sl@0
    78
				{
sl@0
    79
				RDebug::Print(_L("Error %d deleting \"%S\" file.\n"), err, &aFullName);
sl@0
    80
				}
sl@0
    81
			}
sl@0
    82
		fsSession.Close();
sl@0
    83
		}
sl@0
    84
	else
sl@0
    85
		{
sl@0
    86
		RDebug::Print(_L("Error %d connecting file session. File: %S.\n"), err, &aFullName);
sl@0
    87
		}
sl@0
    88
	}
sl@0
    89
sl@0
    90
static void DeleteFiles()
sl@0
    91
	{
sl@0
    92
	::DeleteFile(KTestFile1);
sl@0
    93
	::DeleteFile(KCreTestFile1);
sl@0
    94
	::DeleteFile(KTmpTestFile1);
sl@0
    95
	}
sl@0
    96
sl@0
    97
static TInt GetFileTimeStamp(const TDesC& aFullName, TTime& aTime)
sl@0
    98
	{
sl@0
    99
	RFs fsSession;
sl@0
   100
	TInt err = fsSession.Connect();
sl@0
   101
	if(err == KErrNone)
sl@0
   102
		{
sl@0
   103
		TEntry entry;
sl@0
   104
		if(fsSession.Entry(aFullName, entry) == KErrNone)
sl@0
   105
			{
sl@0
   106
			aTime=entry.iModified;
sl@0
   107
			}
sl@0
   108
		fsSession.Close();
sl@0
   109
		}
sl@0
   110
	else
sl@0
   111
		{
sl@0
   112
		RDebug::Print(_L("Error %d connecting file session. File: %S.\n"), err, &aFullName);
sl@0
   113
		}
sl@0
   114
	return err;
sl@0
   115
	}
sl@0
   116
sl@0
   117
sl@0
   118
///////////////////////////////////////////////////////////////////////////////////////
sl@0
   119
// Test macros and functions.
sl@0
   120
sl@0
   121
static void Check(TInt aValue, TInt aLine)
sl@0
   122
	{
sl@0
   123
	if(!aValue)
sl@0
   124
		{
sl@0
   125
		::DeleteFiles();
sl@0
   126
		TheTest(EFalse, aLine);
sl@0
   127
		}
sl@0
   128
	}
sl@0
   129
static  void Check(TInt aValue, TInt aExpected, TInt aLine)
sl@0
   130
	{
sl@0
   131
	if(aValue != aExpected)
sl@0
   132
		{
sl@0
   133
		::DeleteFiles();
sl@0
   134
		RDebug::Print(_L("*** Expected error: %d, got: %d\r\n"), aExpected, aValue);
sl@0
   135
		TheTest(EFalse, aLine);
sl@0
   136
		}
sl@0
   137
	}
sl@0
   138
#define TEST(arg) ::Check((arg), __LINE__)
sl@0
   139
#define TEST2(aValue, aExpected) ::Check(aValue, aExpected, __LINE__)
sl@0
   140
sl@0
   141
///////////////////////////////////////////////////////////////////////////////////////
sl@0
   142
sl@0
   143
static void CheckTmpFileDeleted()
sl@0
   144
	{
sl@0
   145
	RFs fsSession;
sl@0
   146
	TInt err = fsSession.Connect();
sl@0
   147
	if(err == KErrNone)
sl@0
   148
		{
sl@0
   149
		// Allow time for file to be deleted
sl@0
   150
		User::After(500*1000);
sl@0
   151
		TEST2 (BaflUtils::FileExists (fsSession, KTmpTestFile1), EFalse);
sl@0
   152
		fsSession.Close();
sl@0
   153
		}
sl@0
   154
	else
sl@0
   155
		{
sl@0
   156
		RDebug::Print(_L("Error %d connecting file session.\n"), err);
sl@0
   157
		}
sl@0
   158
	}
sl@0
   159
sl@0
   160
sl@0
   161
///////////////////////////////////////////////////////////////////////////////////////
sl@0
   162
// CTestObserver class
sl@0
   163
sl@0
   164
//It is needed for the creation of CServerRepository object.
sl@0
   165
//Used also to track Get/Set/Create/Delete setting notifications, received
sl@0
   166
//from CServerRepository object.
sl@0
   167
class CTestObserver : public CBase, public MObserver
sl@0
   168
	{
sl@0
   169
public:
sl@0
   170
	CTestObserver();
sl@0
   171
	void Notify(TUint32 aId);
sl@0
   172
sl@0
   173
public:
sl@0
   174
	TBool iAssertNotification;
sl@0
   175
sl@0
   176
	};
sl@0
   177
sl@0
   178
CTestObserver::CTestObserver() :
sl@0
   179
	iAssertNotification(EFalse)
sl@0
   180
	{
sl@0
   181
	}
sl@0
   182
sl@0
   183
void CTestObserver::Notify(TUint32 aId)
sl@0
   184
	{
sl@0
   185
	RDebug::Print(_L("Notification! Id=%d.\n"), aId);
sl@0
   186
	if(iAssertNotification)
sl@0
   187
		{
sl@0
   188
		TEST(0);
sl@0
   189
		}
sl@0
   190
	}
sl@0
   191
sl@0
   192
///////////////////////////////////////////////////////////////////////////////////////
sl@0
   193
//
sl@0
   194
sl@0
   195
//Deletes the global TheObserver object. Sets it to NULL.
sl@0
   196
static void ReleaseObserver(TAny*)
sl@0
   197
	{
sl@0
   198
	delete TheObserver;
sl@0
   199
	TheObserver = NULL;
sl@0
   200
	}
sl@0
   201
sl@0
   202
//Deletes the global TheServerRepository object. Sets it to NULL.
sl@0
   203
static void ReleaseRepository(TAny*)
sl@0
   204
	{
sl@0
   205
	TheServerRepository->Close();
sl@0
   206
	delete TheServerRepository;
sl@0
   207
	TheServerRepository = NULL;
sl@0
   208
	TServerResources::iCacheManager->DisableCache(ETrue);
sl@0
   209
	TServerResources::iObserver->CloseiOpenRepositories();
sl@0
   210
	TServerResources::iCacheManager->EnableCache();
sl@0
   211
	}
sl@0
   212
sl@0
   213
//This function is called from CreateSetting_OOMtestL().
sl@0
   214
//It creates an integer, real or string setting in the repository
sl@0
   215
//with ID=KTestRepositoryId. All this is done in an OOM loop.
sl@0
   216
//The output from this test is: the repository file with ID = KTestRepositoryId
sl@0
   217
//will be copied from Z to C drive;
sl@0
   218
// An integer, real or string setting will be created in that repository.
sl@0
   219
//Changes in the repository will be stored on C drive (in the related repository file)
sl@0
   220
static void DoCreateSettingL(TBool aTrapping,TServerSetting::TType aSettingType)
sl@0
   221
	{
sl@0
   222
	TheObserver = new (ELeave) CTestObserver;
sl@0
   223
	TCleanupItem clnItem1(&ReleaseObserver, NULL);
sl@0
   224
	CleanupStack::PushL(clnItem1);
sl@0
   225
sl@0
   226
	TheServerRepository = new (ELeave) CServerRepository;
sl@0
   227
	TCleanupItem clnItem2(&ReleaseRepository, NULL);
sl@0
   228
	CleanupStack::PushL(clnItem2);
sl@0
   229
sl@0
   230
	// Turn OOM trapping on Repository Open on/off for testing purposes.
sl@0
   231
	TServerResources::iObserver->iTrapOOMOnOpen=aTrapping;
sl@0
   232
	// Open the repository
sl@0
   233
	TheServerRepository->OpenL(KTestRepositoryId, *TheObserver);
sl@0
   234
sl@0
   235
	// All write operations, including CreateL must be done in a transaction
sl@0
   236
	TheServerRepository->StartTransaction(CRepository::EConcurrentReadWriteTransaction);
sl@0
   237
	TheServerRepository->CleanupCancelTransactionPushL();
sl@0
   238
sl@0
   239
	//Create the appropriate setting based on aValueType
sl@0
   240
	switch(aSettingType)
sl@0
   241
		{
sl@0
   242
		case TServerSetting::EInt:
sl@0
   243
			{
sl@0
   244
			// Create setting - fails if value already there (it must not be)
sl@0
   245
			User::LeaveIfError(TheServerRepository->TransactionCreateL(KIntSettingID, KIntSettingVal, NULL));
sl@0
   246
sl@0
   247
			// Try to get the value from the transaction cache
sl@0
   248
			TInt val = 0;
sl@0
   249
			User::LeaveIfError(TheServerRepository->Get(KIntSettingID, val));
sl@0
   250
			TEST2(val,KIntSettingVal);
sl@0
   251
			break;
sl@0
   252
			}
sl@0
   253
sl@0
   254
		case TServerSetting::EReal:
sl@0
   255
			{
sl@0
   256
			// Create setting - fails if value already there (it must not be)
sl@0
   257
			User::LeaveIfError(TheServerRepository->TransactionCreateL(KRealSettingID, KRealSettingVal, NULL));
sl@0
   258
sl@0
   259
			// Try to get the value from the transaction cache
sl@0
   260
			TReal val = 0;
sl@0
   261
			User::LeaveIfError(TheServerRepository->Get(KRealSettingID, val));
sl@0
   262
			TEST(val == KRealSettingVal);
sl@0
   263
			break;
sl@0
   264
			}
sl@0
   265
sl@0
   266
		case TServerSetting::EString:
sl@0
   267
			{
sl@0
   268
			// Create setting - fails if value already there (it must not be)
sl@0
   269
			User::LeaveIfError(TheServerRepository->TransactionCreateL(KStringSettingID, KStringSettingVal, NULL));
sl@0
   270
sl@0
   271
			// Try to get the value from the transaction cache
sl@0
   272
			TBuf8<16> bufVal(KEmptyString);
sl@0
   273
			User::LeaveIfError(TheServerRepository->Get(KStringSettingID, bufVal));
sl@0
   274
			TEST(bufVal == KStringSettingVal);
sl@0
   275
			break;
sl@0
   276
			}
sl@0
   277
		default:
sl@0
   278
			break;
sl@0
   279
		}
sl@0
   280
sl@0
   281
	// Commit the changes
sl@0
   282
	CleanupStack::Pop();
sl@0
   283
	TUint32 keyInfo;
sl@0
   284
	User::LeaveIfError(TheServerRepository->CommitTransaction(keyInfo));
sl@0
   285
sl@0
   286
	TServerResources::iObserver->iTrapOOMOnOpen=EFalse;
sl@0
   287
	// Close the repository
sl@0
   288
	TheServerRepository->Close();
sl@0
   289
sl@0
   290
	CleanupStack::PopAndDestroy(2);//TheObserver & TheServerRepository
sl@0
   291
	}
sl@0
   292
sl@0
   293
//This function is called from SetSetting_OOMtest().
sl@0
   294
//It expects that the previous test (CreateSetting_OOMtestL()) had completed successfully
sl@0
   295
//and there is a repository file (ID=0x10) on drive C and there is an integer setting
sl@0
   296
//in that repository (ID=KIntSettingID, Value=KIntSettingVal).
sl@0
   297
//The test will try to change the setting value in a transaction. Because the test is
sl@0
   298
//executed in an OOM loop, CommitChangesL() - will fail when called by SetL, leaving the
sl@0
   299
//repository content inconsistent - the in-memory repository content will be deleted
sl@0
   300
//(no settings), but in  the file the setting will exist and its value will still be KIntSettingVal.
sl@0
   301
static void DoSetIntSettingL()
sl@0
   302
	{
sl@0
   303
	TEST(TheObserver != NULL);
sl@0
   304
	TEST(TheServerRepository != NULL);
sl@0
   305
	// Check if the setting is there - and has (or is restored to) the original value
sl@0
   306
	TInt val = 0;
sl@0
   307
	User::LeaveIfError(TheServerRepository->Get(KIntSettingID, val));
sl@0
   308
	TEST2(val,KIntSettingVal);
sl@0
   309
sl@0
   310
	// Change the setting value
sl@0
   311
	// All write operations, including CreateL must be done in a transaction
sl@0
   312
	TheServerRepository->StartTransaction(CRepository::EConcurrentReadWriteTransaction);
sl@0
   313
	TheServerRepository->CleanupCancelTransactionPushL();
sl@0
   314
	// Change the setting value
sl@0
   315
	User::LeaveIfError(TheServerRepository->TransactionSetL(KIntSettingID, KIntSettingVal + 1));
sl@0
   316
sl@0
   317
	// Check if the setting is there - it should be in transaction cache now
sl@0
   318
	User::LeaveIfError(TheServerRepository->Get(KIntSettingID, val));
sl@0
   319
	TEST2(val,(KIntSettingVal + 1));
sl@0
   320
sl@0
   321
	// Commit the changes
sl@0
   322
	CleanupStack::Pop();
sl@0
   323
	TUint32 keyInfo;
sl@0
   324
	User::LeaveIfError(TheServerRepository->CommitTransaction(keyInfo));
sl@0
   325
sl@0
   326
	// Check if the setting is there. The transaction was successful so the new value
sl@0
   327
	// should be returned.
sl@0
   328
	val = 0;
sl@0
   329
	User::LeaveIfError(TheServerRepository->Get(KIntSettingID, val));
sl@0
   330
	TEST2(val,(KIntSettingVal + 1));
sl@0
   331
	}
sl@0
   332
sl@0
   333
//This function is called from SetSetting_OOMtest().
sl@0
   334
//It expects that the previous test (CreateSetting_OOMtestL()) had completed successfully
sl@0
   335
//and there is a repository file (ID=0x10) on drive C and there is an rea; setting
sl@0
   336
//in that repository (ID=KRealSettingID, Value=KRealSettingVal).
sl@0
   337
//The test will try to change the setting value in a transaction. Because the test is
sl@0
   338
//executed in an OOM loop, CommitChangesL() - will fail when called by SetL, leaving the
sl@0
   339
//repository content inconsistent - the in-memory repository content will be deleted
sl@0
   340
//(no settings), but in  the file the setting will exist and its value will still be KRealSettingVal.
sl@0
   341
static void DoSetRealSettingL()
sl@0
   342
	{
sl@0
   343
	TEST(TheObserver != NULL);
sl@0
   344
	TEST(TheServerRepository != NULL);
sl@0
   345
	// Check if the setting is there - and has (or is restored to) the original value
sl@0
   346
	TReal val = 0;
sl@0
   347
	User::LeaveIfError(TheServerRepository->Get(KRealSettingID, val));
sl@0
   348
	TEST(val == KRealSettingVal);
sl@0
   349
sl@0
   350
	// Change the setting value
sl@0
   351
	// All write operations, including CreateL must be done in a transaction
sl@0
   352
	TheServerRepository->StartTransaction(CRepository::EConcurrentReadWriteTransaction);
sl@0
   353
	TheServerRepository->CleanupCancelTransactionPushL();
sl@0
   354
	// Change the setting value
sl@0
   355
	User::LeaveIfError(TheServerRepository->TransactionSetL(KRealSettingID, KRealSettingVal2));
sl@0
   356
sl@0
   357
	// Check if the setting is there - it should be in transaction cache now
sl@0
   358
	User::LeaveIfError(TheServerRepository->Get(KRealSettingID, val));
sl@0
   359
	TEST(val == KRealSettingVal2);
sl@0
   360
sl@0
   361
	// Commit the changes
sl@0
   362
	CleanupStack::Pop();
sl@0
   363
	TUint32 keyInfo;
sl@0
   364
	User::LeaveIfError(TheServerRepository->CommitTransaction(keyInfo));
sl@0
   365
sl@0
   366
	// Check if the setting is there. The transaction was successful so the new value
sl@0
   367
	// should be returned.
sl@0
   368
	val = 0;
sl@0
   369
	User::LeaveIfError(TheServerRepository->Get(KRealSettingID, val));
sl@0
   370
	TEST(val == KRealSettingVal2);
sl@0
   371
	}
sl@0
   372
sl@0
   373
//This function is called from SetSetting_OOMtest().
sl@0
   374
//It expects that the previous test (CreateSetting_OOMtestL()) had completed successfully
sl@0
   375
//and there is a repository file (ID=0x10) on drive C and there is an string setting
sl@0
   376
//in that repository (ID=KStringSettingID, Value=KStringSettingVal).
sl@0
   377
//The test will try to change the setting value in a transaction. Because the test is
sl@0
   378
//executed in an OOM loop, CommitChangesL() - will fail when called by SetL, leaving the
sl@0
   379
//repository content inconsistent - the in-memory repository content will be deleted
sl@0
   380
//(no settings), but in  the file the setting will exist and its value will still
sl@0
   381
//be KStringSettingVal.
sl@0
   382
static void DoSetStringSettingL()
sl@0
   383
	{
sl@0
   384
	TEST(TheObserver != NULL);
sl@0
   385
	TEST(TheServerRepository != NULL);
sl@0
   386
sl@0
   387
	// Check if the setting is there - and has (or is restored to) the original value
sl@0
   388
	TBuf8<16> bufVal(KEmptyString);
sl@0
   389
	User::LeaveIfError(TheServerRepository->Get(KStringSettingID, bufVal));
sl@0
   390
	TEST(bufVal == KStringSettingVal);
sl@0
   391
sl@0
   392
	// Change the setting value
sl@0
   393
	// All write operations, including CreateL must be done in a transaction
sl@0
   394
	TheServerRepository->StartTransaction(CRepository::EConcurrentReadWriteTransaction);
sl@0
   395
	TheServerRepository->CleanupCancelTransactionPushL();
sl@0
   396
sl@0
   397
	// Change the setting value
sl@0
   398
	User::LeaveIfError(TheServerRepository->TransactionSetL(KStringSettingID, KStringSettingVal2));
sl@0
   399
sl@0
   400
	// Check if the setting is there - it should be in transaction cache now
sl@0
   401
	User::LeaveIfError(TheServerRepository->Get(KStringSettingID, bufVal));
sl@0
   402
	TEST(bufVal == KStringSettingVal2);
sl@0
   403
	// Commit the changes
sl@0
   404
	CleanupStack::Pop();
sl@0
   405
	TUint32 keyInfo;
sl@0
   406
	User::LeaveIfError(TheServerRepository->CommitTransaction(keyInfo));
sl@0
   407
sl@0
   408
	// Check if the setting is there. The transaction was successful so the new value
sl@0
   409
	// should be returned.
sl@0
   410
	bufVal = KEmptyString;
sl@0
   411
	User::LeaveIfError(TheServerRepository->Get(KStringSettingID, bufVal));
sl@0
   412
	TEST(bufVal == KStringSettingVal2);
sl@0
   413
	}
sl@0
   414
sl@0
   415
//This function is called from SetSetting_OOMtest()in the case where previous
sl@0
   416
//DoSetSettingL() call failed with KErrNoMemory, leaving the repository object
sl@0
   417
//in inconsisten state - the in-memory presentation of the repository differs from
sl@0
   418
//the repository file content. The function will try to repair the repository,
sl@0
   419
//as it would have to happen in a real situation. Then it will check that
sl@0
   420
//the repository content is consistent, which means the tested setting value should
sl@0
   421
//be the same as it was before DoSetSettingL() call.
sl@0
   422
static void DoRecoverRepositoryL(TServerSetting::TType aSettingType)
sl@0
   423
	{
sl@0
   424
	//Repair the repository as it happens in a real situation
sl@0
   425
	//AccessL calls RestoreConsistencyL() indirectly
sl@0
   426
	//RestoreConsistencyL() is called in the production code from the session object,
sl@0
   427
	//ServiceL() implementation. So, it will be called before any other repository call
sl@0
   428
	//and it should restore the repository content - it should repair the repository
sl@0
   429
	//consitency.
sl@0
   430
	TServerResources::iObserver->AccessL(KTestRepositoryId);
sl@0
   431
	//Check if the setting is there - the old, pre-transactional value should be
sl@0
   432
	//in-memory now.
sl@0
   433
	switch(aSettingType)
sl@0
   434
		{
sl@0
   435
		case TServerSetting::EInt:
sl@0
   436
			{
sl@0
   437
			TInt val = 0;
sl@0
   438
			User::LeaveIfError(TheServerRepository->Get(KIntSettingID, val));
sl@0
   439
			TEST2(val,KIntSettingVal);
sl@0
   440
			break;
sl@0
   441
			}
sl@0
   442
sl@0
   443
		case TServerSetting::EReal:
sl@0
   444
			{
sl@0
   445
			TReal val = 0;
sl@0
   446
			User::LeaveIfError(TheServerRepository->Get(KRealSettingID, val));
sl@0
   447
			TEST(val == KRealSettingVal);
sl@0
   448
			break;
sl@0
   449
			}
sl@0
   450
sl@0
   451
		case TServerSetting::EString:
sl@0
   452
			{
sl@0
   453
			TBuf8<16> bufVal(KEmptyString);
sl@0
   454
			User::LeaveIfError(TheServerRepository->Get(KStringSettingID, bufVal));
sl@0
   455
			TEST(bufVal == KStringSettingVal);
sl@0
   456
			break;
sl@0
   457
			}
sl@0
   458
		default:
sl@0
   459
			break;
sl@0
   460
		}
sl@0
   461
	}
sl@0
   462
sl@0
   463
//Inits TServerResources object.
sl@0
   464
//It should be called once, before the creation of CServerRepository object.
sl@0
   465
static void InitEnvL()
sl@0
   466
	{
sl@0
   467
	TServerResources::InitialiseL();
sl@0
   468
sl@0
   469
	delete TServerResources::iDataDirectory; TServerResources::iDataDirectory = NULL;
sl@0
   470
	delete TServerResources::iRomDirectory; TServerResources::iRomDirectory = NULL;
sl@0
   471
	delete TServerResources::iInstallDirectory; TServerResources::iInstallDirectory = NULL;
sl@0
   472
sl@0
   473
	TServerResources::iDataDirectory = HBufC::NewL(KMaxFileName);
sl@0
   474
	TServerResources::iRomDirectory = HBufC::NewL(KMaxFileName);
sl@0
   475
	TServerResources::iInstallDirectory = HBufC::NewL(KMaxFileName);
sl@0
   476
sl@0
   477
	TBuf<32> uidName;
sl@0
   478
	uidName.Format(_L("%08X"), KCenRepServerUID.iUid);
sl@0
   479
sl@0
   480
	TPtr ptr1(TServerResources::iDataDirectory->Des());
sl@0
   481
	ptr1.Append(_L("C:\\PRIVATE\\"));
sl@0
   482
	ptr1.Append(uidName);
sl@0
   483
	ptr1.Append(_L("\\PERSISTS\\"));
sl@0
   484
sl@0
   485
	TPtr ptr2(TServerResources::iRomDirectory->Des());
sl@0
   486
	ptr2.Append(_L("Z:\\PRIVATE\\"));
sl@0
   487
	ptr2.Append(uidName);
sl@0
   488
	ptr2.Append(_L("\\"));
sl@0
   489
sl@0
   490
	TPtr ptr3(TServerResources::iInstallDirectory->Des());
sl@0
   491
	ptr3.Append(_L("C:\\PRIVATE\\"));
sl@0
   492
	ptr3.Append(uidName);
sl@0
   493
	ptr3.Append(_L("\\"));
sl@0
   494
	}
sl@0
   495
sl@0
   496
//Destroys TServerResources object.
sl@0
   497
//It should be called after the CServerRepository object was destroyed.
sl@0
   498
static void ReleaseEnv()
sl@0
   499
	{
sl@0
   500
	TServerResources::Close();
sl@0
   501
	}
sl@0
   502
sl@0
   503
//Creates global TheServerRepository and TheObserver objects
sl@0
   504
//It is used in SetSetting_OOMtest()
sl@0
   505
static void InitEnv2L()
sl@0
   506
	{
sl@0
   507
	TheObserver = new CTestObserver;
sl@0
   508
	TEST(TheObserver != NULL);
sl@0
   509
sl@0
   510
	TheServerRepository = new CServerRepository;
sl@0
   511
	TEST(TheServerRepository != NULL);
sl@0
   512
sl@0
   513
	TheServerRepository->OpenL(KTestRepositoryId, *TheObserver);
sl@0
   514
	}
sl@0
   515
sl@0
   516
//Destroys global TheServerRepository and TheObserver objects
sl@0
   517
//It is used in SetSetting_OOMtest()
sl@0
   518
static void ReleaseEnv2()
sl@0
   519
	{
sl@0
   520
	::ReleaseRepository(NULL);
sl@0
   521
	::ReleaseObserver(NULL);
sl@0
   522
	}
sl@0
   523
sl@0
   524
//Central repository test: it creates a setting in OOM conditions.
sl@0
   525
static void CreateSetting_OOMtestL(TBool aTrapping,TServerSetting::TType aSettingType)
sl@0
   526
	{
sl@0
   527
	TInt err;
sl@0
   528
	TRAP(err, ::InitEnvL());
sl@0
   529
	TEST2(err, KErrNone);
sl@0
   530
	for(TInt count=1;;++count)
sl@0
   531
		{
sl@0
   532
		CleanupCDriveL();
sl@0
   533
		__UHEAP_FAILNEXT(count);
sl@0
   534
		__UHEAP_MARK;
sl@0
   535
sl@0
   536
		TRAP(err, ::DoCreateSettingL(aTrapping,aSettingType));
sl@0
   537
sl@0
   538
		TServerResources::iOwnerIdLookUpTable.Reset();
sl@0
   539
		TServerResources::iObserver->Reset();
sl@0
   540
sl@0
   541
		__UHEAP_MARKEND;
sl@0
   542
sl@0
   543
		if(err == KErrNone)
sl@0
   544
			{
sl@0
   545
			RDebug::Print(_L("The test succeeded at heap failure rate=%d.\n"), count);
sl@0
   546
			break;
sl@0
   547
			}
sl@0
   548
		else if(err != KErrNoMemory)
sl@0
   549
			{
sl@0
   550
			TEST2(err, KErrNone);
sl@0
   551
			}
sl@0
   552
		}
sl@0
   553
sl@0
   554
	::ReleaseEnv();
sl@0
   555
	__UHEAP_RESET;
sl@0
   556
	}
sl@0
   557
sl@0
   558
//Central repository test: it sets new setting value in OOM conditions.
sl@0
   559
//Then ComitChangesL() is called to store the new setting's value in the repository
sl@0
   560
//file. Because of the simulated OOM conditions ComitChangesL() can fail, leaving the
sl@0
   561
//repository in inconsistent state.
sl@0
   562
//DoRecoverRepositoryL() is called to repair the repository and restore the consistency.
sl@0
   563
static void SetSetting_OOMtest(TServerSetting::TType aSettingType)
sl@0
   564
	{
sl@0
   565
	TTime before;
sl@0
   566
	TTime after;
sl@0
   567
	// Check that cre file exists and get modification time
sl@0
   568
	TEST2 (GetFileTimeStamp(KCreTestFile1, before), KErrNone);
sl@0
   569
	after=before;
sl@0
   570
sl@0
   571
	TInt err;
sl@0
   572
	TRAP(err, ::InitEnvL());
sl@0
   573
	TEST2(err, KErrNone);
sl@0
   574
	TRAP(err, ::InitEnv2L());
sl@0
   575
	TEST2(err, KErrNone);	
sl@0
   576
	for(TInt count=1;;++count)
sl@0
   577
		{
sl@0
   578
		__UHEAP_FAILNEXT(count);
sl@0
   579
		__UHEAP_MARK;
sl@0
   580
sl@0
   581
	switch(aSettingType)
sl@0
   582
		{
sl@0
   583
		case TServerSetting::EInt:
sl@0
   584
			{
sl@0
   585
			TRAP(err, ::DoSetIntSettingL());
sl@0
   586
			break;
sl@0
   587
			}
sl@0
   588
		case TServerSetting::EReal:
sl@0
   589
			{
sl@0
   590
			TRAP(err, ::DoSetRealSettingL());
sl@0
   591
			break;
sl@0
   592
			}
sl@0
   593
		case TServerSetting::EString:
sl@0
   594
			{
sl@0
   595
			TRAP(err, ::DoSetStringSettingL());
sl@0
   596
			break;
sl@0
   597
			}
sl@0
   598
		default:
sl@0
   599
			break;
sl@0
   600
		}
sl@0
   601
sl@0
   602
		// Get timestamp of cre file. It is now possible for the commit to
sl@0
   603
		// fail with KErrNoMemory after a successful write. If filestamp
sl@0
   604
		// changes, the cre file has been written and the test is complete
sl@0
   605
		TEST2 (GetFileTimeStamp(KCreTestFile1, after), KErrNone);
sl@0
   606
		if(before!=after)
sl@0
   607
			{
sl@0
   608
			err=KErrNone;
sl@0
   609
			}
sl@0
   610
sl@0
   611
		if(err == KErrNoMemory)
sl@0
   612
			{
sl@0
   613
			__UHEAP_MARKEND;
sl@0
   614
			//DoSetSettingL() call failed (so the transaction). Try to recover.
sl@0
   615
			TInt err2;
sl@0
   616
			TRAP(err2, ::DoRecoverRepositoryL(aSettingType));
sl@0
   617
			TEST2(err2, KErrNone);
sl@0
   618
			}
sl@0
   619
		else if(err == KErrNone)
sl@0
   620
			{
sl@0
   621
			//The heap cannot be checked at this point because memory may have been
sl@0
   622
			//allocated which must be freed by a call to ReleaseEnv2.
sl@0
   623
			//It is checked outside the loop instead
sl@0
   624
			RDebug::Print(_L("The test succeeded at heap failure rate=%d.\n"), count);
sl@0
   625
			break;
sl@0
   626
			}
sl@0
   627
		else
sl@0
   628
			{
sl@0
   629
			__UHEAP_MARKEND;
sl@0
   630
			TEST2(err, KErrNone);
sl@0
   631
			}
sl@0
   632
		// Check that tmp file generated by attempted commit is deleted
sl@0
   633
		// by RestoreConsistencyL() in DoRecoverRepositoryL().
sl@0
   634
		::CheckTmpFileDeleted();
sl@0
   635
		}
sl@0
   636
	// Check that no tmp file is left over
sl@0
   637
	::ReleaseEnv2();
sl@0
   638
	::ReleaseEnv();
sl@0
   639
	//This __UHEAP_MARKEND checks the heap after the call to DoSetxSettings
sl@0
   640
	//succeeds as calls to CommitTRansactions can allocate memory which must be freed
sl@0
   641
	//by calling ReleaseEnv2
sl@0
   642
	__UHEAP_MARKEND;
sl@0
   643
	__UHEAP_RESET;
sl@0
   644
	}
sl@0
   645
sl@0
   646
///////////////////////////////////////////////////////////////////////////////////////
sl@0
   647
sl@0
   648
void RunTestsL(TServerSetting::TType aSettingType)
sl@0
   649
	{
sl@0
   650
sl@0
   651
	TheTest.Next(_L("Create setting - OOM test - OOM Trapping off"));
sl@0
   652
	::CreateSetting_OOMtestL(EFalse,aSettingType);
sl@0
   653
sl@0
   654
	TheTest.Next(_L("Create setting - OOM test - OOM Trapping on"));
sl@0
   655
	::CreateSetting_OOMtestL(ETrue,aSettingType);
sl@0
   656
sl@0
   657
	TheTest.Next(_L("Set setting - transaction - OOM test"));
sl@0
   658
	::SetSetting_OOMtest(aSettingType);
sl@0
   659
sl@0
   660
	}
sl@0
   661
sl@0
   662
static void MainL()
sl@0
   663
	{
sl@0
   664
	// create and install the active scheduler we need for the cache manager in TServerResources::InitialiseL
sl@0
   665
	CActiveScheduler* cacheManagerAS=new(ELeave) CActiveScheduler;
sl@0
   666
	CleanupStack::PushL(cacheManagerAS);
sl@0
   667
	CActiveScheduler::Install(cacheManagerAS);
sl@0
   668
sl@0
   669
	::DeleteFiles();
sl@0
   670
sl@0
   671
	TheTest.Start(_L(" @SYMTestCaseID:SYSLIB-CENTRALREPOSITORY-LEGACY-T_CENREPTRANS-0001 Int Setting Tests "));
sl@0
   672
	RunTestsL(TServerSetting::EInt);
sl@0
   673
sl@0
   674
	TheTest.Next(_L("Real Setting Tests"));
sl@0
   675
	RunTestsL(TServerSetting::EReal);
sl@0
   676
sl@0
   677
	TheTest.Next(_L("String Setting Tests"));
sl@0
   678
	RunTestsL(TServerSetting::EString);
sl@0
   679
sl@0
   680
	TheTest.End();
sl@0
   681
	TheTest.Close();
sl@0
   682
sl@0
   683
	CleanupStack::PopAndDestroy(cacheManagerAS);
sl@0
   684
	}
sl@0
   685
sl@0
   686
TInt E32Main()
sl@0
   687
	{
sl@0
   688
	__UHEAP_MARK;
sl@0
   689
sl@0
   690
	CTrapCleanup* tc = CTrapCleanup::New();
sl@0
   691
	TEST(tc != NULL);
sl@0
   692
sl@0
   693
	TheTest.Title();
sl@0
   694
	TRAPD(err, ::MainL());
sl@0
   695
	TEST2(err, KErrNone);
sl@0
   696
sl@0
   697
	::DeleteFiles();
sl@0
   698
sl@0
   699
	delete tc;
sl@0
   700
sl@0
   701
	__UHEAP_MARKEND;
sl@0
   702
sl@0
   703
	User::Heap().Check();
sl@0
   704
	return KErrNone;
sl@0
   705
	}