os/persistentdata/persistentstorage/store/TFILE/t_storfserr.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) 1998-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 <s32file.h>
sl@0
    17
#include <e32test.h>
sl@0
    18
sl@0
    19
LOCAL_D RTest test(_L("t_storfserr"));
sl@0
    20
sl@0
    21
#ifdef _DEBUG
sl@0
    22
sl@0
    23
LOCAL_D CTrapCleanup* TheTrapCleanup = NULL;
sl@0
    24
LOCAL_D RFs TheFs;
sl@0
    25
sl@0
    26
const TInt KTestCleanupStack=0x20;
sl@0
    27
const TPtrC KTestDir=_L("\\STOR-TST\\T_FSERR\\");
sl@0
    28
const TPtrC FileNameA=_L("A.DAT");
sl@0
    29
const TPtrC FileNameB=_L("B.DAT");
sl@0
    30
sl@0
    31
const TPtrC des2(_S("22"),2);
sl@0
    32
const TPtrC des3(_S("333"),3);
sl@0
    33
const TPtrC des4(_S("4444"),4);
sl@0
    34
const TPtrC des5(_S("55555"),5);
sl@0
    35
LOCAL_D CFileStore* store = NULL;
sl@0
    36
RStoreWriteStream out;
sl@0
    37
RStoreReadStream in;
sl@0
    38
sl@0
    39
LOCAL_C void setupCleanup()
sl@0
    40
    {// Initialise the cleanup stack
sl@0
    41
	TheTrapCleanup=CTrapCleanup::New();
sl@0
    42
	test(TheTrapCleanup!=NULL);
sl@0
    43
	TRAPD(r,\
sl@0
    44
		{\
sl@0
    45
		for (TInt i=KTestCleanupStack;i>0;i--)\
sl@0
    46
			CleanupStack::PushL((TAny*)0);\
sl@0
    47
		CleanupStack::Pop(KTestCleanupStack);\
sl@0
    48
		});
sl@0
    49
	test(r==KErrNone);
sl@0
    50
	}
sl@0
    51
sl@0
    52
LOCAL_C void setupTestDirectory()
sl@0
    53
    {// Prepare the test directory.
sl@0
    54
	TInt r=TheFs.MkDirAll(KTestDir);
sl@0
    55
	test(r==KErrNone||r==KErrAlreadyExists);
sl@0
    56
	r=TheFs.SetSessionPath(KTestDir);
sl@0
    57
	test(r==KErrNone);
sl@0
    58
	}
sl@0
    59
sl@0
    60
LOCAL_D void AlterStoreL(TInt aFail)
sl@0
    61
	{
sl@0
    62
	RStoreWriteStream out2;
sl@0
    63
	RStoreWriteStream out3;
sl@0
    64
	RStoreWriteStream out4;
sl@0
    65
	RStoreReadStream in;
sl@0
    66
	TheFs.SetErrorCondition(KErrNotReady,aFail);
sl@0
    67
	TStreamId id0 = out.CreateLC(*store);
sl@0
    68
	out << _L("012345678901234567890123456789012345678901234567890123456789");
sl@0
    69
	out.CommitL();
sl@0
    70
	CleanupStack::PopAndDestroy();
sl@0
    71
sl@0
    72
	TStreamId id2 = out.CreateLC(*store);
sl@0
    73
	out.CommitL();
sl@0
    74
	CleanupStack::PopAndDestroy();
sl@0
    75
sl@0
    76
	TStreamId id3 = out.CreateLC(*store);
sl@0
    77
	out.CommitL();
sl@0
    78
	CleanupStack::PopAndDestroy();
sl@0
    79
sl@0
    80
	TStreamId id4 = out.CreateLC(*store);
sl@0
    81
	out << _L("mum");
sl@0
    82
	out.CommitL();
sl@0
    83
	CleanupStack::PopAndDestroy();
sl@0
    84
sl@0
    85
	out.ReplaceLC(*store,store->Root());
sl@0
    86
	out << id2;
sl@0
    87
	out << id3;
sl@0
    88
	out << id4;
sl@0
    89
	out << KNullStreamId;
sl@0
    90
	out << KNullStreamId;
sl@0
    91
	out.CommitL();
sl@0
    92
	CleanupStack::PopAndDestroy();
sl@0
    93
sl@0
    94
	in.OpenLC(*store,store->Root());// use the root for in and out streams
sl@0
    95
	out.ReplaceLC(*store,store->Root());
sl@0
    96
	out.WriteL(in);
sl@0
    97
	out.CommitL();
sl@0
    98
	CleanupStack::PopAndDestroy(2);
sl@0
    99
sl@0
   100
	out.ReplaceLC(*store,store->Root());// swap the order
sl@0
   101
	in.OpenLC(*store,store->Root());
sl@0
   102
	out.WriteL(in);
sl@0
   103
	out << KNullStreamId;
sl@0
   104
	out.CommitL();
sl@0
   105
	CleanupStack::PopAndDestroy(2);
sl@0
   106
sl@0
   107
	store->CommitL();
sl@0
   108
sl@0
   109
	in.OpenLC(*store,store->Root());
sl@0
   110
	TStreamId idX,idZ;
sl@0
   111
	in >> idX;
sl@0
   112
	in >> idX;
sl@0
   113
	in >> idZ;// id4 "mum"
sl@0
   114
	CleanupStack::PopAndDestroy();
sl@0
   115
	out.OpenLC(*store,idZ);
sl@0
   116
	in.OpenLC(*store,idZ);
sl@0
   117
	out2.OpenLC(*store,idZ);
sl@0
   118
	out3.OpenLC(*store,idZ);
sl@0
   119
	out4.OpenLC(*store,idZ);
sl@0
   120
	out4.WriteL(in);
sl@0
   121
	out.CommitL();
sl@0
   122
	CleanupStack::PopAndDestroy(5);
sl@0
   123
sl@0
   124
	store->CompactL();
sl@0
   125
	store->CommitL();
sl@0
   126
sl@0
   127
	store->DeleteL(id0);
sl@0
   128
	store->CommitL();
sl@0
   129
sl@0
   130
	store->CompactL();
sl@0
   131
	store->CommitL();
sl@0
   132
	}
sl@0
   133
sl@0
   134
enum TFileQoS
sl@0
   135
	{
sl@0
   136
	ESimple			= 0,//File, "write byte" is an atomic operation
sl@0
   137
	EBlockAtomic	= 1,//File, "block write" is an atomic operation
sl@0
   138
	ETransactional	= 2,//Transactional file system.
sl@0
   139
	ELastQoSType
sl@0
   140
	};
sl@0
   141
sl@0
   142
const TInt KFailPoints[ELastQoSType][2] =
sl@0
   143
	{
sl@0
   144
	// non-transactional file system
sl@0
   145
		{
sl@0
   146
		5, 54
sl@0
   147
		},
sl@0
   148
	// atomic "block write"
sl@0
   149
		{
sl@0
   150
		3, 38
sl@0
   151
		},
sl@0
   152
	// transactional file system
sl@0
   153
		{
sl@0
   154
		3, 38
sl@0
   155
		}
sl@0
   156
	};
sl@0
   157
sl@0
   158
//The function returns true, if the file system guarantees atomic "block write" operations on aDriveNo.
sl@0
   159
TBool IsBlockAtomic(TInt aDriveNo)
sl@0
   160
	{
sl@0
   161
	__ASSERT_DEBUG(aDriveNo >= EDriveA && aDriveNo <= EDriveZ, User::Invariant());
sl@0
   162
sl@0
   163
	TVolumeIOParamInfo volInfo;
sl@0
   164
	TInt err = TheFs.VolumeIOParam(aDriveNo, volInfo);
sl@0
   165
	//If VolumeIOParam() succeeds, the media block size is >= 512 bytes and the media block size is power of two - report
sl@0
   166
	//that the media supports atomic "block write" operations.
sl@0
   167
	const TInt KDefaultMediaBlockSize = 512;
sl@0
   168
	return err == KErrNone && volInfo.iBlockSize >= KDefaultMediaBlockSize && (volInfo.iBlockSize & (volInfo.iBlockSize - 1)) == 0;
sl@0
   169
	}
sl@0
   170
sl@0
   171
TFileQoS MediaType(const TDesC& aFilename)
sl@0
   172
	{
sl@0
   173
	TParse p;
sl@0
   174
	TheFs.Parse(aFilename, p);
sl@0
   175
	TDriveInfo dinfo;
sl@0
   176
	TheFs.Drive(dinfo, TDriveUnit(p.Drive()));
sl@0
   177
	TBool transactional = dinfo.iDriveAtt & KDriveAttTransaction;
sl@0
   178
	if(transactional)
sl@0
   179
		{
sl@0
   180
		return ETransactional;
sl@0
   181
		}
sl@0
   182
	if(IsBlockAtomic(TDriveUnit(p.Drive())))
sl@0
   183
		{
sl@0
   184
		return EBlockAtomic;
sl@0
   185
		}
sl@0
   186
	return ESimple;
sl@0
   187
	}
sl@0
   188
/**
sl@0
   189
@SYMTestCaseID          SYSLIB-STORE-CT-1160
sl@0
   190
@SYMTestCaseDesc	    File server test
sl@0
   191
@SYMTestPriority 	    High
sl@0
   192
@SYMTestActions  	    Tests for failure on call to file server
sl@0
   193
@SYMTestExpectedResults Test must not fail
sl@0
   194
@SYMREQ                 REQ0000
sl@0
   195
*/
sl@0
   196
LOCAL_D void FailOnEveryFileServerCallL()
sl@0
   197
	{
sl@0
   198
	TheFs.Delete(FileNameA);
sl@0
   199
sl@0
   200
	const TFileQoS KMediaType = MediaType(FileNameA);
sl@0
   201
	_LIT(KSimpleText, "Simple");
sl@0
   202
	_LIT(KAtomicText, "Atomic block \"write\"");
sl@0
   203
	_LIT(KTransactionalText, "Transactional");
sl@0
   204
	const TPtrC mediaType[ELastQoSType] = {KSimpleText(), KAtomicText(), KTransactionalText()};
sl@0
   205
	test.Printf(_L("Media type: %S\r\n"), &mediaType[KMediaType]);
sl@0
   206
sl@0
   207
	const TInt KStoreCommitted = KFailPoints[KMediaType][0];
sl@0
   208
	const TInt KLastFail = KFailPoints[KMediaType][1];
sl@0
   209
sl@0
   210
	const TInt KError = KErrNotReady;
sl@0
   211
	const TInt KRootIdSizeBeforeStoreCommit = 0;
sl@0
   212
	const TInt KRootIdSizeAfterStoreCommit = 6*sizeof(TStreamId);
sl@0
   213
		
sl@0
   214
	for (TInt fail=1;fail<=KLastFail+5;++fail)
sl@0
   215
		{
sl@0
   216
		store=CPermanentFileStore::ReplaceLC(TheFs,FileNameA,EFileWrite|EFileRead);
sl@0
   217
		store->SetTypeL(KPermanentFileStoreLayoutUid);
sl@0
   218
		TStreamId rootId = store->ExtendL();
sl@0
   219
		store->SetRootL(rootId);
sl@0
   220
		store->CommitL();
sl@0
   221
		TRAPD(r,AlterStoreL(fail));
sl@0
   222
		TheFs.SetErrorCondition(KErrNone);
sl@0
   223
		if (fail<KLastFail)
sl@0
   224
			test(r==KError);
sl@0
   225
		else
sl@0
   226
			test(r==KErrNone);
sl@0
   227
		TRAP(r,store->RevertL());
sl@0
   228
		if (r==KErrNotReady)
sl@0
   229
			{
sl@0
   230
			CleanupStack::PopAndDestroy();
sl@0
   231
			store=CPermanentFileStore::OpenLC(TheFs,FileNameA,EFileWrite|EFileRead);
sl@0
   232
			}
sl@0
   233
		else
sl@0
   234
			test(r==KErrNone);
sl@0
   235
		CleanupStack::PopAndDestroy();
sl@0
   236
//
sl@0
   237
		// open read only
sl@0
   238
		store=CPermanentFileStore::OpenLC(TheFs,FileNameA,EFileRead);
sl@0
   239
		RStoreReadStream in;
sl@0
   240
		in.OpenLC(*store,store->Root());
sl@0
   241
		TInt size=in.Source()->SizeL();
sl@0
   242
		if (fail<KStoreCommitted)
sl@0
   243
			test(size==KRootIdSizeBeforeStoreCommit);
sl@0
   244
		else //fail>=KStoreCommitted
sl@0
   245
			test(size==KRootIdSizeAfterStoreCommit);
sl@0
   246
		CleanupStack::PopAndDestroy(2);
sl@0
   247
//
sl@0
   248
		store=CPermanentFileStore::OpenLC(TheFs,FileNameA,EFileRead|EFileWrite);
sl@0
   249
		store->ExtendL();
sl@0
   250
		store->RevertL();
sl@0
   251
		in.OpenLC(*store,store->Root());
sl@0
   252
		size=in.Source()->SizeL();
sl@0
   253
		if (fail<KStoreCommitted)
sl@0
   254
			test(size==KRootIdSizeBeforeStoreCommit);
sl@0
   255
		else //fail>=KStoreCommitted
sl@0
   256
			test(size==KRootIdSizeAfterStoreCommit);
sl@0
   257
		CleanupStack::PopAndDestroy(2);
sl@0
   258
		}
sl@0
   259
	}
sl@0
   260
sl@0
   261
LOCAL_D void InitialseStoreWithDataL()
sl@0
   262
	{
sl@0
   263
	TheFs.Delete(FileNameB);
sl@0
   264
	store=CPermanentFileStore::CreateLC(TheFs,FileNameB,EFileWrite|EFileRead);
sl@0
   265
	store->SetTypeL(KPermanentFileStoreLayoutUid);
sl@0
   266
	TStreamId rootId = store->ExtendL();
sl@0
   267
	store->SetRootL(rootId);
sl@0
   268
	store->CommitL();
sl@0
   269
	CleanupStack::PopAndDestroy();
sl@0
   270
sl@0
   271
	store=CPermanentFileStore::OpenLC(TheFs,FileNameB,EFileWrite|EFileRead);
sl@0
   272
	TStreamId id2 = out.CreateLC(*store);
sl@0
   273
	out << des2;
sl@0
   274
	out.CommitL();
sl@0
   275
	CleanupStack::PopAndDestroy();
sl@0
   276
sl@0
   277
	TStreamId id3 = out.CreateLC(*store);
sl@0
   278
	out << des3;
sl@0
   279
	out.CommitL();
sl@0
   280
	CleanupStack::PopAndDestroy();
sl@0
   281
sl@0
   282
	out.ReplaceLC(*store,store->Root());
sl@0
   283
	out << id2;
sl@0
   284
	out << id3;
sl@0
   285
	out.CommitL();
sl@0
   286
	CleanupStack::PopAndDestroy();// out
sl@0
   287
sl@0
   288
	store->CommitL();
sl@0
   289
	CleanupStack::PopAndDestroy();// store
sl@0
   290
	}
sl@0
   291
sl@0
   292
LOCAL_D void AlterStoreDuringFileServerErrorL(TInt aError)
sl@0
   293
	{
sl@0
   294
	store=CPermanentFileStore::OpenLC(TheFs,FileNameB,EFileWrite|EFileRead);
sl@0
   295
	in.OpenLC(*store,store->Root());
sl@0
   296
	TStreamId id2;
sl@0
   297
	TStreamId id3;
sl@0
   298
	in >> id2;
sl@0
   299
	in >> id3;
sl@0
   300
	CleanupStack::PopAndDestroy();// in
sl@0
   301
sl@0
   302
	out.ReplaceLC(*store,id2);
sl@0
   303
	out << des4;
sl@0
   304
	out.CommitL();
sl@0
   305
	CleanupStack::PopAndDestroy();// out
sl@0
   306
	TheFs.SetErrorCondition(aError);
sl@0
   307
	out.ReplaceLC(*store,id3);
sl@0
   308
	out << des5;
sl@0
   309
	out.CommitL();
sl@0
   310
	CleanupStack::PopAndDestroy();// out
sl@0
   311
sl@0
   312
	store->CommitL();
sl@0
   313
	CleanupStack::PopAndDestroy();// store
sl@0
   314
	}
sl@0
   315
sl@0
   316
LOCAL_D void TestStreamDataL(TInt aError)
sl@0
   317
	{
sl@0
   318
	store=CPermanentFileStore::OpenLC(TheFs,FileNameB,EFileWrite|EFileRead);
sl@0
   319
	in.OpenLC(*store,store->Root());
sl@0
   320
	TStreamId id2;
sl@0
   321
	TStreamId id3;
sl@0
   322
	in >> id2;
sl@0
   323
	in >> id3;
sl@0
   324
	CleanupStack::PopAndDestroy();// in
sl@0
   325
sl@0
   326
	TBuf<16> buf;
sl@0
   327
sl@0
   328
	in.OpenLC(*store,id2);
sl@0
   329
	in >> buf;
sl@0
   330
	if (aError==KErrNone)
sl@0
   331
		test(buf==_L("4444"));// store committed
sl@0
   332
	else
sl@0
   333
		test(buf==_L("22"));// store reverted
sl@0
   334
sl@0
   335
	CleanupStack::PopAndDestroy();// in
sl@0
   336
sl@0
   337
	in.OpenLC(*store,id3);
sl@0
   338
	in >> buf;
sl@0
   339
	if (aError==KErrNone)
sl@0
   340
		test(buf==_L("55555"));// store committed
sl@0
   341
	else
sl@0
   342
		test(buf==_L("333"));// store reverted
sl@0
   343
sl@0
   344
	CleanupStack::PopAndDestroy();// in
sl@0
   345
sl@0
   346
	CleanupStack::PopAndDestroy();// store
sl@0
   347
	}
sl@0
   348
sl@0
   349
LOCAL_D void ResetStreamDataL()
sl@0
   350
	{
sl@0
   351
	store=CPermanentFileStore::OpenLC(TheFs,FileNameB,EFileWrite|EFileRead);
sl@0
   352
	in.OpenLC(*store,store->Root());
sl@0
   353
	TStreamId id2;
sl@0
   354
	TStreamId id3;
sl@0
   355
	in >> id2;
sl@0
   356
	in >> id3;
sl@0
   357
	CleanupStack::PopAndDestroy();// in
sl@0
   358
sl@0
   359
	out.ReplaceLC(*store,id2);
sl@0
   360
	out << des2;
sl@0
   361
	out.CommitL();
sl@0
   362
	CleanupStack::PopAndDestroy();// out
sl@0
   363
sl@0
   364
	out.ReplaceLC(*store,id3);
sl@0
   365
	out << des3;
sl@0
   366
	out.CommitL();
sl@0
   367
	CleanupStack::PopAndDestroy();// out
sl@0
   368
sl@0
   369
	store->CommitL();
sl@0
   370
	CleanupStack::PopAndDestroy();// store
sl@0
   371
	}
sl@0
   372
/**
sl@0
   373
@SYMTestCaseID          SYSLIB-STORE-CT-1161
sl@0
   374
@SYMTestCaseDesc	    Tests for all errors on file server
sl@0
   375
@SYMTestPriority 	    High
sl@0
   376
@SYMTestActions  	    Tests for stream data to a store
sl@0
   377
@SYMTestExpectedResults Test must not fail
sl@0
   378
@SYMREQ                 REQ0000
sl@0
   379
*/
sl@0
   380
LOCAL_D void TestAllFsErrorL()
sl@0
   381
	{
sl@0
   382
	InitialseStoreWithDataL();
sl@0
   383
	for (TInt error=0; error >= KErrDied; --error)
sl@0
   384
		{
sl@0
   385
		TRAPD(r,AlterStoreDuringFileServerErrorL(error));
sl@0
   386
		TheFs.SetErrorCondition(KErrNone);
sl@0
   387
		if (r!=KErrNone)
sl@0
   388
			{
sl@0
   389
			store = CPermanentFileStore::OpenLC(TheFs,FileNameB,EFileWrite|EFileRead);
sl@0
   390
			store->RevertL();
sl@0
   391
			CleanupStack::PopAndDestroy();// store
sl@0
   392
			}
sl@0
   393
		TestStreamDataL(error);
sl@0
   394
		ResetStreamDataL();
sl@0
   395
		}
sl@0
   396
	}
sl@0
   397
sl@0
   398
void RunTests()
sl@0
   399
	{
sl@0
   400
	setupTestDirectory();
sl@0
   401
	__UHEAP_MARK;
sl@0
   402
//
sl@0
   403
	test.Start(_L(" @SYMTestCaseID:SYSLIB-STORE-CT-1160 Random failure on every file server call "));
sl@0
   404
	TRAPD(r,FailOnEveryFileServerCallL());
sl@0
   405
	test(r==KErrNone);
sl@0
   406
	test.Next(_L(" @SYMTestCaseID:SYSLIB-STORE-CT-1161 Alter STORE during file server error "));
sl@0
   407
	TRAP(r,TestAllFsErrorL());
sl@0
   408
	test(r==KErrNone);
sl@0
   409
	test.End();
sl@0
   410
sl@0
   411
	TheFs.Delete(FileNameA);
sl@0
   412
	TheFs.Delete(FileNameB);
sl@0
   413
//
sl@0
   414
	__UHEAP_MARKEND;
sl@0
   415
	}
sl@0
   416
sl@0
   417
#endif	// _DEBUG
sl@0
   418
sl@0
   419
GLDEF_C TInt E32Main()
sl@0
   420
    {// Test permanent file store.
sl@0
   421
	test.Title();
sl@0
   422
#ifdef _DEBUG
sl@0
   423
	TInt r=TheFs.Connect();
sl@0
   424
	test(r==KErrNone);
sl@0
   425
	setupCleanup();
sl@0
   426
	RunTests();
sl@0
   427
	delete TheTrapCleanup;
sl@0
   428
	TheFs.Close();
sl@0
   429
#else
sl@0
   430
	test.Start(_L("The tests are not valid in release mode"));
sl@0
   431
	test.End();
sl@0
   432
#endif
sl@0
   433
	test.Close();
sl@0
   434
	return 0;
sl@0
   435
    }
sl@0
   436