os/persistentdata/persistentstorage/store/TSTOR/t_stortoc.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
//
sl@0
    15
sl@0
    16
#include <e32test.h>
sl@0
    17
#include <s32file.h>
sl@0
    18
#include <s32fileiter.h>
sl@0
    19
sl@0
    20
RTest TheTest(_L("t_storetoc test"));
sl@0
    21
RFs TheFs;
sl@0
    22
sl@0
    23
TBuf<KMaxFileName> TheFileName;
sl@0
    24
sl@0
    25
CPermanentFileStore* TheStore = NULL;
sl@0
    26
sl@0
    27
void DeleteTestFiles()
sl@0
    28
	{
sl@0
    29
	delete TheStore;
sl@0
    30
	TheStore = NULL;
sl@0
    31
	(void)TheFs.Delete(TheFileName);
sl@0
    32
	}
sl@0
    33
sl@0
    34
///////////////////////////////////////////////////////////////////////////////////////
sl@0
    35
///////////////////////////////////////////////////////////////////////////////////////
sl@0
    36
//Test macros and functions
sl@0
    37
void Check(TInt aValue, TInt aLine)
sl@0
    38
	{
sl@0
    39
	if(!aValue)
sl@0
    40
		{
sl@0
    41
		DeleteTestFiles();
sl@0
    42
		TheTest(EFalse, aLine);
sl@0
    43
		}
sl@0
    44
	}
sl@0
    45
void Check(TInt aValue, TInt aExpected, TInt aLine)
sl@0
    46
	{
sl@0
    47
	if(aValue != aExpected)
sl@0
    48
		{
sl@0
    49
		DeleteTestFiles();
sl@0
    50
		TheTest.Printf(_L("*** Expected error: %d, got: %d\r\n"), aExpected, aValue);
sl@0
    51
		TheTest(EFalse, aLine);
sl@0
    52
		}
sl@0
    53
	}
sl@0
    54
#define TEST(arg) ::Check((arg), __LINE__)
sl@0
    55
#define TEST2(aValue, aExpected) ::Check(aValue, aExpected, __LINE__)
sl@0
    56
sl@0
    57
///////////////////////////////////////////////////////////////////////////////////////
sl@0
    58
sl@0
    59
void CreateTestEnv(TInt aDriveNo)
sl@0
    60
    {
sl@0
    61
    TChar ch;
sl@0
    62
	TInt err = TheFs.DriveToChar(aDriveNo, ch);
sl@0
    63
	TEST2(err, KErrNone);
sl@0
    64
    TBuf<2> drvName;
sl@0
    65
    drvName.SetLength(2);
sl@0
    66
    drvName[0] = ch;
sl@0
    67
    drvName[1] = TChar(':');
sl@0
    68
sl@0
    69
	_LIT(KTestFile, "c:\\stor-tst\\t_storetoc.dat");
sl@0
    70
sl@0
    71
    TParse parse;
sl@0
    72
    parse.Set(drvName, &KTestFile, 0);
sl@0
    73
	const TDesC& dbFilePath = parse.FullName();
sl@0
    74
	TheFileName.Copy(dbFilePath);
sl@0
    75
sl@0
    76
	err = TheFs.MkDir(parse.DriveAndPath());
sl@0
    77
	TEST(err == KErrNone || err == KErrAlreadyExists);
sl@0
    78
	}
sl@0
    79
sl@0
    80
///////////////////////////////////////////////////////////////////////////////////////
sl@0
    81
//////////////////////              TTocTestBase        ///////////////////////////////
sl@0
    82
///////////////////////////////////////////////////////////////////////////////////////
sl@0
    83
//Base TOC test class.
sl@0
    84
//It offers the folowing functions: Prepare(), Test() and Verify().
sl@0
    85
//The test algorithm is:
sl@0
    86
//  =================================
sl@0
    87
//  void TestFunc(TTocTestBase& aTestObj)
sl@0
    88
//  {
sl@0
    89
//	  aTestObj.Prepare();
sl@0
    90
//    [simulate file system error];
sl@0
    91
//	  TInt err = aTestObj.Test();
sl@0
    92
//    [clear file system error];
sl@0
    93
//	  aTestObj.Verify(err);
sl@0
    94
//  }
sl@0
    95
//  =================================
sl@0
    96
//The class has two private pure virtual methods, which have to be overriden in derived classes:
sl@0
    97
//  DoPrepareL() - instance specific test preparation;
sl@0
    98
//  DoVerify()   - instance specific test verification;
sl@0
    99
class TTocTestBase
sl@0
   100
	{
sl@0
   101
public:
sl@0
   102
	TTocTestBase()
sl@0
   103
		{
sl@0
   104
		TheStore = NULL;
sl@0
   105
		}
sl@0
   106
	void Prepare()
sl@0
   107
		{
sl@0
   108
		TEST(TheStore == NULL);
sl@0
   109
		TRAPD(err, DoPrepareL());
sl@0
   110
		TEST2(err, KErrNone);
sl@0
   111
		}
sl@0
   112
	TInt Test()
sl@0
   113
		{
sl@0
   114
		TEST(TheStore != NULL);
sl@0
   115
		TRAPD(err, {TheStore->CompactL(); TheStore->CommitL();});
sl@0
   116
		delete TheStore;
sl@0
   117
		TheStore = NULL;
sl@0
   118
		return err;
sl@0
   119
		}
sl@0
   120
	void Verify(TInt aTestErr)
sl@0
   121
		{
sl@0
   122
		TEST(TheStore == NULL);
sl@0
   123
		TRAPD(err, DoVerifyL(aTestErr));
sl@0
   124
		TEST2(err, KErrNone);
sl@0
   125
		}
sl@0
   126
sl@0
   127
protected:
sl@0
   128
	TStreamId WriteNewStreamL(const TDesC8& aStreamData)
sl@0
   129
		{
sl@0
   130
		TEST(TheStore != NULL);
sl@0
   131
		RStoreWriteStream stream;
sl@0
   132
		TStreamId streamId = stream.CreateLC(*TheStore);
sl@0
   133
		stream.WriteL(aStreamData);
sl@0
   134
		stream.CommitL();
sl@0
   135
		TheStore->CommitL();
sl@0
   136
		CleanupStack::PopAndDestroy(&stream);
sl@0
   137
		return streamId;
sl@0
   138
		}
sl@0
   139
sl@0
   140
private:
sl@0
   141
	virtual void DoPrepareL() = 0;
sl@0
   142
	virtual void DoVerifyL(TInt aTestErr) = 0;
sl@0
   143
sl@0
   144
	};
sl@0
   145
sl@0
   146
///////////////////////////////////////////////////////////////////////////////////////
sl@0
   147
//////////////////////                TBaseTocEntryTest        ////////////////////////
sl@0
   148
///////////////////////////////////////////////////////////////////////////////////////
sl@0
   149
sl@0
   150
//Do not change/shift the next lines, because "new line/space" characters will be inserted in the string!
sl@0
   151
_LIT8(KStreamData1, "-01-ABCD-02-ABCD-03-ABCD-04-ABCD-05-ABCD-06-ABCD-07-ABCD-08-ABCD");
sl@0
   152
_LIT8(KStreamData1_2, "ABCDEFGH-41-ABCD-42-ABCD-43-A\
sl@0
   153
-33-ABCD-34-ABCD-35-ABCD-36-ABCD-37-ABCD-38-ABCD-39-ABCD-40-ABCD\
sl@0
   154
-25-ABCD-26-ABCD-27-ABCD-28-ABCD-29-ABCD-30-ABCD-31-ABCD-32-ABCD\
sl@0
   155
-17-ABCD-18-ABCD-19-ABCD-20-ABCD-21-ABCD-22-ABCD-23-ABCD-24-ABCD\
sl@0
   156
-09-ABCD-10-ABCD-11-ABCD-12-ABCD-13-ABCD-14-ABCD-15-ABCD-16-ABCD");
sl@0
   157
_LIT8(KStreamData1_3, "0123456789");
sl@0
   158
sl@0
   159
//"Base TOC entry" test.
sl@0
   160
//When a stream is relocated, its base TOC entry needs 2 "file write" (for "atomic block write" media) operations, because
sl@0
   161
//the TOC entry is split over a block boundary.
sl@0
   162
//This test class uses a set of test data, which puts the TOC entry of the relocated stream at position
sl@0
   163
//510 in the file (TOC entry length - 5 bytes).
sl@0
   164
class TBaseTocEntryTest : public TTocTestBase
sl@0
   165
	{
sl@0
   166
public:
sl@0
   167
	TBaseTocEntryTest() :
sl@0
   168
		iStreamId1(KNullStreamIdValue),
sl@0
   169
		iStreamId2(KNullStreamIdValue),
sl@0
   170
		iStreamId3(KNullStreamIdValue)
sl@0
   171
		{
sl@0
   172
		}
sl@0
   173
sl@0
   174
private:
sl@0
   175
	void DoPrepareL()
sl@0
   176
		{
sl@0
   177
		TheStore = CPermanentFileStore::ReplaceLC(TheFs, TheFileName, EFileWrite);
sl@0
   178
		TheStore->SetTypeL(TheStore->Layout());
sl@0
   179
		TheStore->CommitL();
sl@0
   180
sl@0
   181
		iStreamId1 = WriteNewStreamL(KStreamData1);
sl@0
   182
		TheStore->SetRootL(iStreamId1);
sl@0
   183
		TheStore->CommitL();
sl@0
   184
		iStreamId2 = WriteNewStreamL(KStreamData1_2);
sl@0
   185
		iStreamId3 = WriteNewStreamL(KStreamData1_3);
sl@0
   186
sl@0
   187
		TheStore->DeleteL(iStreamId2);
sl@0
   188
		TheStore->CommitL();
sl@0
   189
sl@0
   190
		CleanupStack::Pop(TheStore);
sl@0
   191
		}
sl@0
   192
sl@0
   193
	virtual void DoVerifyL(TInt)
sl@0
   194
		{
sl@0
   195
		TEST(TheStore == NULL);
sl@0
   196
		TheStore = CPermanentFileStore::OpenLC(TheFs, TheFileName, EFileWrite);
sl@0
   197
		RPermanentFileStoreIter iter;
sl@0
   198
		iter.ResetLC(*TheStore);
sl@0
   199
		//Stream 1
sl@0
   200
		TStreamId streamId = iter.NextL();
sl@0
   201
		TEST(streamId == iStreamId1);
sl@0
   202
		RStoreReadStream strm;
sl@0
   203
		strm.OpenLC(*TheStore, streamId);
sl@0
   204
		TEST2(strm.Source()->SizeL(), KStreamData1().Length());
sl@0
   205
		TBuf8<sizeof(KStreamData1)> buf;
sl@0
   206
		strm.ReadL(buf, KStreamData1().Length());
sl@0
   207
		TEST2(Mem::Compare(buf.Ptr(), buf.Length(), KStreamData1().Ptr(), KStreamData1().Length()), 0);
sl@0
   208
		CleanupStack::PopAndDestroy(&strm);
sl@0
   209
		//Stream 2
sl@0
   210
		streamId = iter.NextL();
sl@0
   211
		TEST(streamId == iStreamId3);
sl@0
   212
		strm.OpenLC(*TheStore, streamId);
sl@0
   213
		TEST2(strm.Source()->SizeL(), KStreamData1_3().Length());
sl@0
   214
		strm.ReadL(buf, KStreamData1_3().Length());
sl@0
   215
		TEST2(Mem::Compare(buf.Ptr(), buf.Length(), KStreamData1_3().Ptr(), KStreamData1_3().Length()), 0);
sl@0
   216
		CleanupStack::PopAndDestroy(&strm);
sl@0
   217
		//No more streams
sl@0
   218
		streamId = iter.NextL();
sl@0
   219
		TEST(streamId == KNullStreamIdValue);
sl@0
   220
		//Cleanup
sl@0
   221
		CleanupStack::PopAndDestroy(&iter);
sl@0
   222
		CleanupStack::PopAndDestroy(TheStore);
sl@0
   223
		TheStore = NULL;
sl@0
   224
		}
sl@0
   225
sl@0
   226
private:
sl@0
   227
	TStreamId iStreamId1;
sl@0
   228
	TStreamId iStreamId2;
sl@0
   229
	TStreamId iStreamId3;
sl@0
   230
sl@0
   231
	};
sl@0
   232
sl@0
   233
///////////////////////////////////////////////////////////////////////////////////////
sl@0
   234
//////////////////////              TDeltaTocEntryTest        /////////////////////////
sl@0
   235
///////////////////////////////////////////////////////////////////////////////////////
sl@0
   236
sl@0
   237
//Do not change/shift the next lines, because "new line/space" characters will be inserted in the string!
sl@0
   238
_LIT8(KStreamData2, "\
sl@0
   239
-01-ABCD-02-ABCD-03-ABCD-04-ABCD-05-ABCD-06-ABCD-07-ABCD-08-ABCD\
sl@0
   240
-09-ABCD-10-ABCD-11-ABCD-12");
sl@0
   241
_LIT8(KStreamData2_2, "ABC");
sl@0
   242
_LIT8(KStreamData2_3, "012345678");
sl@0
   243
sl@0
   244
//"Delta TOC entry" test
sl@0
   245
//When a stream is relocated, its delta TOC entry needs 2 "file write" (for "atomic block write" media) operations, because
sl@0
   246
//the TOC entry is split over a block boundary.
sl@0
   247
//This test class uses a set of test data, which puts the TOC entry of the relocated stream at position
sl@0
   248
//507 in the file (TOC entry length - 8 bytes), so the last 4 bytes (stream offset) are split.
sl@0
   249
class TDeltaTocEntryTest : public TTocTestBase
sl@0
   250
	{
sl@0
   251
public:
sl@0
   252
	TDeltaTocEntryTest() :
sl@0
   253
		iStreamId1(KNullStreamIdValue),
sl@0
   254
		iStreamId2(KNullStreamIdValue),
sl@0
   255
		iStreamId3(KNullStreamIdValue),
sl@0
   256
		iStreamId4(KNullStreamIdValue),
sl@0
   257
		iStreamId5(KNullStreamIdValue)
sl@0
   258
		{
sl@0
   259
		}
sl@0
   260
sl@0
   261
protected:
sl@0
   262
	virtual void DoPrepareL()
sl@0
   263
		{
sl@0
   264
		TheStore = CPermanentFileStore::ReplaceLC(TheFs, TheFileName, EFileWrite);
sl@0
   265
		TheStore->SetTypeL(TheStore->Layout());
sl@0
   266
		TheStore->CommitL();
sl@0
   267
sl@0
   268
		iStreamId1 = WriteNewStreamL(KStreamData2);
sl@0
   269
		TheStore->SetRootL(iStreamId1);
sl@0
   270
		TheStore->CommitL();
sl@0
   271
		iStreamId2 = WriteNewStreamL(KStreamData2);
sl@0
   272
		iStreamId3 = WriteNewStreamL(KStreamData2().Left(KStreamData2().Length() - 1));
sl@0
   273
		iStreamId4 = WriteNewStreamL(KStreamData2_2);
sl@0
   274
		iStreamId5 = WriteNewStreamL(KStreamData2_3);
sl@0
   275
sl@0
   276
		TheStore->DeleteL(iStreamId4);
sl@0
   277
		TheStore->CommitL();
sl@0
   278
sl@0
   279
		CleanupStack::Pop(TheStore);
sl@0
   280
		}
sl@0
   281
sl@0
   282
	virtual void DoVerifyL(TInt)
sl@0
   283
		{
sl@0
   284
		TEST(TheStore == NULL);
sl@0
   285
		TheStore = CPermanentFileStore::OpenLC(TheFs, TheFileName, EFileWrite);
sl@0
   286
		RPermanentFileStoreIter iter;
sl@0
   287
		iter.ResetLC(*TheStore);
sl@0
   288
		//Stream 1
sl@0
   289
		TStreamId streamId = iter.NextL();
sl@0
   290
		TEST(streamId == iStreamId1);
sl@0
   291
		RStoreReadStream strm;
sl@0
   292
		strm.OpenLC(*TheStore, streamId);
sl@0
   293
		TEST2(strm.Source()->SizeL(), KStreamData2().Length());
sl@0
   294
		TBuf8<sizeof(KStreamData2)> buf;
sl@0
   295
		strm.ReadL(buf, KStreamData2().Length());
sl@0
   296
		TEST2(Mem::Compare(buf.Ptr(), buf.Length(), KStreamData2().Ptr(), KStreamData2().Length()), 0);
sl@0
   297
		CleanupStack::PopAndDestroy(&strm);
sl@0
   298
		//Stream 2
sl@0
   299
		streamId = iter.NextL();
sl@0
   300
		TEST(streamId == iStreamId2);
sl@0
   301
		strm.OpenLC(*TheStore, streamId);
sl@0
   302
		TEST2(strm.Source()->SizeL(), KStreamData2().Length());
sl@0
   303
		strm.ReadL(buf, KStreamData2().Length());
sl@0
   304
		TEST2(Mem::Compare(buf.Ptr(), buf.Length(), KStreamData2().Ptr(), KStreamData2().Length()), 0);
sl@0
   305
		CleanupStack::PopAndDestroy(&strm);
sl@0
   306
		//Stream 3
sl@0
   307
		streamId = iter.NextL();
sl@0
   308
		TEST(streamId == iStreamId3);
sl@0
   309
		strm.OpenLC(*TheStore, streamId);
sl@0
   310
		TEST2(strm.Source()->SizeL(), KStreamData2().Length() - 1);
sl@0
   311
		strm.ReadL(buf, KStreamData2().Length() - 1);
sl@0
   312
		TEST2(Mem::Compare(buf.Ptr(), buf.Length(), KStreamData2().Ptr(), KStreamData2().Length() - 1), 0);
sl@0
   313
		CleanupStack::PopAndDestroy(&strm);
sl@0
   314
		//Stream 4
sl@0
   315
		streamId = iter.NextL();
sl@0
   316
		TEST(streamId == iStreamId5);
sl@0
   317
		strm.OpenLC(*TheStore, streamId);
sl@0
   318
		TEST2(strm.Source()->SizeL(), KStreamData2_3().Length());
sl@0
   319
		strm.ReadL(buf, KStreamData2_3().Length());
sl@0
   320
		TEST2(Mem::Compare(buf.Ptr(), buf.Length(), KStreamData2_3().Ptr(), KStreamData2_3().Length()), 0);
sl@0
   321
		CleanupStack::PopAndDestroy(&strm);
sl@0
   322
		//No more streams
sl@0
   323
		streamId = iter.NextL();
sl@0
   324
		TEST(streamId == KNullStreamIdValue);
sl@0
   325
		//Cleanup
sl@0
   326
		CleanupStack::PopAndDestroy(&iter);
sl@0
   327
		CleanupStack::PopAndDestroy(TheStore);
sl@0
   328
		TheStore = NULL;
sl@0
   329
		}
sl@0
   330
sl@0
   331
private:
sl@0
   332
	TStreamId iStreamId1;
sl@0
   333
	TStreamId iStreamId2;
sl@0
   334
	TStreamId iStreamId3;
sl@0
   335
	TStreamId iStreamId4;
sl@0
   336
	TStreamId iStreamId5;
sl@0
   337
sl@0
   338
	};
sl@0
   339
sl@0
   340
///////////////////////////////////////////////////////////////////////////////////////
sl@0
   341
//////////////////////              TBaseTocReferenceTest        //////////////////////
sl@0
   342
///////////////////////////////////////////////////////////////////////////////////////
sl@0
   343
sl@0
   344
//Do not change/shift the next lines, because "new line/space" characters will be inserted in the string!
sl@0
   345
_LIT8(KStreamData3, "\
sl@0
   346
-01-ABCD-02-ABCD-03-ABCD-04-ABCD-05-ABCD-06-ABCD-07-ABCD-08-ABCD\
sl@0
   347
-09-ABCD-10-ABCD-11-ABCD-12-ABCD");
sl@0
   348
_LIT8(KStreamData3_2, "ABCDEFGH");
sl@0
   349
_LIT8(KStreamData3_3, "0123456789");
sl@0
   350
sl@0
   351
//"Base TOC reference" test.
sl@0
   352
//When the base TOC is relocated, the delta TOC reference to the base TOC has to be updated with
sl@0
   353
// 2 "file write" operations (for "atomic block write" media), because the reference is split over a block boundary.
sl@0
   354
//This test class uses a set of test data, which puts the delta TOC reference to the base TOC
sl@0
   355
//at position 511 in the file (reference length - 4 bytes)
sl@0
   356
class TBaseTocReferenceTest : public TTocTestBase
sl@0
   357
	{
sl@0
   358
public:
sl@0
   359
	TBaseTocReferenceTest() :
sl@0
   360
		iStreamId1(KNullStreamIdValue),
sl@0
   361
		iStreamId2(KNullStreamIdValue),
sl@0
   362
		iStreamId3(KNullStreamIdValue),
sl@0
   363
		iStreamId4(KNullStreamIdValue),
sl@0
   364
		iStreamId5(KNullStreamIdValue)
sl@0
   365
		{
sl@0
   366
		}
sl@0
   367
sl@0
   368
protected:
sl@0
   369
	virtual void DoPrepareL()
sl@0
   370
		{
sl@0
   371
		TheStore = CPermanentFileStore::ReplaceLC(TheFs, TheFileName, EFileWrite);
sl@0
   372
		TheStore->SetTypeL(TheStore->Layout());
sl@0
   373
		TheStore->CommitL();
sl@0
   374
sl@0
   375
		iStreamId1 = WriteNewStreamL(KStreamData3);
sl@0
   376
		TheStore->SetRootL(iStreamId1);
sl@0
   377
		TheStore->CommitL();
sl@0
   378
		iStreamId2 = WriteNewStreamL(KStreamData3);
sl@0
   379
		iStreamId3 = WriteNewStreamL(KStreamData3().Left(KStreamData3().Length() - 3));
sl@0
   380
		iStreamId4 = WriteNewStreamL(KStreamData3_2);
sl@0
   381
		iStreamId5 = WriteNewStreamL(KStreamData3_3);
sl@0
   382
sl@0
   383
		TheStore->DeleteL(iStreamId4);
sl@0
   384
		TheStore->CommitL();
sl@0
   385
sl@0
   386
		CleanupStack::Pop(TheStore);
sl@0
   387
		}
sl@0
   388
sl@0
   389
	virtual void DoVerifyL(TInt)
sl@0
   390
		{
sl@0
   391
		TEST(TheStore == NULL);
sl@0
   392
		TheStore = CPermanentFileStore::OpenLC(TheFs, TheFileName, EFileWrite);
sl@0
   393
		RPermanentFileStoreIter iter;
sl@0
   394
		iter.ResetLC(*TheStore);
sl@0
   395
		//Stream 1
sl@0
   396
		TStreamId streamId = iter.NextL();
sl@0
   397
		TEST(streamId == iStreamId1);
sl@0
   398
		RStoreReadStream strm;
sl@0
   399
		strm.OpenLC(*TheStore, streamId);
sl@0
   400
		TEST2(strm.Source()->SizeL(), KStreamData3().Length());
sl@0
   401
		TBuf8<sizeof(KStreamData3)> buf;
sl@0
   402
		strm.ReadL(buf, KStreamData3().Length());
sl@0
   403
		TEST2(Mem::Compare(buf.Ptr(), buf.Length(), KStreamData3().Ptr(), KStreamData3().Length()), 0);
sl@0
   404
		CleanupStack::PopAndDestroy(&strm);
sl@0
   405
		//Stream 2
sl@0
   406
		streamId = iter.NextL();
sl@0
   407
		TEST(streamId == iStreamId2);
sl@0
   408
		strm.OpenLC(*TheStore, streamId);
sl@0
   409
		TEST2(strm.Source()->SizeL(), KStreamData3().Length());
sl@0
   410
		strm.ReadL(buf, KStreamData3().Length());
sl@0
   411
		TEST2(Mem::Compare(buf.Ptr(), buf.Length(), KStreamData3().Ptr(), KStreamData3().Length()), 0);
sl@0
   412
		CleanupStack::PopAndDestroy(&strm);
sl@0
   413
		//Stream 3
sl@0
   414
		streamId = iter.NextL();
sl@0
   415
		TEST(streamId == iStreamId3);
sl@0
   416
		strm.OpenLC(*TheStore, streamId);
sl@0
   417
		TEST2(strm.Source()->SizeL(), KStreamData3().Length() - 3);
sl@0
   418
		strm.ReadL(buf, KStreamData3().Length() - 3);
sl@0
   419
		TEST2(Mem::Compare(buf.Ptr(), buf.Length(), KStreamData3().Ptr(), KStreamData3().Length() - 3), 0);
sl@0
   420
		CleanupStack::PopAndDestroy(&strm);
sl@0
   421
		//Stream 4
sl@0
   422
		streamId = iter.NextL();
sl@0
   423
		TEST(streamId == iStreamId5);
sl@0
   424
		strm.OpenLC(*TheStore, streamId);
sl@0
   425
		TEST2(strm.Source()->SizeL(), KStreamData3_3().Length());
sl@0
   426
		strm.ReadL(buf, KStreamData3_3().Length());
sl@0
   427
		TEST2(Mem::Compare(buf.Ptr(), buf.Length(), KStreamData3_3().Ptr(), KStreamData3_3().Length()), 0);
sl@0
   428
		CleanupStack::PopAndDestroy(&strm);
sl@0
   429
		//No more streams
sl@0
   430
		streamId = iter.NextL();
sl@0
   431
		TEST(streamId == KNullStreamIdValue);
sl@0
   432
		//Cleanup
sl@0
   433
		CleanupStack::PopAndDestroy(&iter);
sl@0
   434
		CleanupStack::PopAndDestroy(TheStore);
sl@0
   435
		TheStore = NULL;
sl@0
   436
		}
sl@0
   437
sl@0
   438
private:
sl@0
   439
	TStreamId iStreamId1;
sl@0
   440
	TStreamId iStreamId2;
sl@0
   441
	TStreamId iStreamId3;
sl@0
   442
	TStreamId iStreamId4;
sl@0
   443
	TStreamId iStreamId5;
sl@0
   444
sl@0
   445
	};
sl@0
   446
sl@0
   447
///////////////////////////////////////////////////////////////////////////////////////
sl@0
   448
///////////////////////////////////////////////////////////////////////////////////////
sl@0
   449
sl@0
   450
void TocTest(TTocTestBase& aTestObj)
sl@0
   451
	{
sl@0
   452
	TInt err = KErrGeneral;
sl@0
   453
	for(TInt cnt=1;err!=KErrNone;++cnt)
sl@0
   454
		{
sl@0
   455
		TheTest.Printf(_L("%d\r"), cnt);
sl@0
   456
		aTestObj.Prepare();
sl@0
   457
		(void)TheFs.SetErrorCondition(KErrGeneral, cnt);
sl@0
   458
		err = aTestObj.Test();
sl@0
   459
		(void)TheFs.SetErrorCondition(KErrNone);
sl@0
   460
		aTestObj.Verify(err);
sl@0
   461
		}
sl@0
   462
	TheTest.Printf(_L("\n"));
sl@0
   463
	TEST2(err, KErrNone);
sl@0
   464
	aTestObj.Verify(err);
sl@0
   465
	}
sl@0
   466
sl@0
   467
/**
sl@0
   468
@SYMTestCaseID          SYSLIB-STORE-CT-3481
sl@0
   469
@SYMTestCaseDesc        The test is performed on media, where atomic "block write" operations are guaranteed.
sl@0
   470
						In a "file I/O error" simulation loop:
sl@0
   471
						 - Creates a CPermanentStoreFile object, stores there carefully selected set of test data,
sl@0
   472
						   such that a base TOC entry is split over the block boundary;
sl@0
   473
						 - Compacts the store;
sl@0
   474
						 - Commits;
sl@0
   475
@SYMTestPriority        High
sl@0
   476
@SYMTestActions         Creates a CPermanentStoreFile object, stores there carefully selected set of test data,
sl@0
   477
						such that a base TOC entry is split over the block boundary. When the store is compacted
sl@0
   478
						and the used media supports atomic "block write" operations, the STORE will update
sl@0
   479
						the base TOC entry, which requires a "file write" operation crossing the block boundary.
sl@0
   480
@SYMTestExpectedResults The test should not fail or panic.
sl@0
   481
@SYMREQ                 REQ0000
sl@0
   482
*/
sl@0
   483
void BaseTocEntryTest()
sl@0
   484
	{
sl@0
   485
	TBaseTocEntryTest baseTocEntryTestObj;
sl@0
   486
	TocTest(baseTocEntryTestObj);
sl@0
   487
	}
sl@0
   488
sl@0
   489
/**
sl@0
   490
@SYMTestCaseID          SYSLIB-STORE-CT-3482
sl@0
   491
@SYMTestCaseDesc        The test is performed on media, where atomic "block write" operations are guaranteed.
sl@0
   492
						In a "file I/O error" simulation loop:
sl@0
   493
						 - Creates a CPermanentStoreFile object, stores there carefully selected set of test data,
sl@0
   494
						   such that a delta TOC entry is split over the block boundary;
sl@0
   495
						 - Compacts the store;
sl@0
   496
						 - Commits;
sl@0
   497
@SYMTestPriority        High
sl@0
   498
@SYMTestActions         Creates a CPermanentStoreFile object, stores there carefully selected set of test data,
sl@0
   499
						such that a delta TOC entry is split over the block boundary. When the store is compacted
sl@0
   500
						and the used media supports atomic "block write" operations, the STORE will update
sl@0
   501
						the delta TOC entry, which requires a "file write" operation crossing the block boundary.
sl@0
   502
@SYMTestExpectedResults The test should not fail or panic.
sl@0
   503
@SYMREQ                 REQ0000
sl@0
   504
*/
sl@0
   505
void DeltaTocEntryTest()
sl@0
   506
	{
sl@0
   507
	TDeltaTocEntryTest deltaTocEntryTestObj;
sl@0
   508
	TocTest(deltaTocEntryTestObj);
sl@0
   509
	}
sl@0
   510
sl@0
   511
/**
sl@0
   512
@SYMTestCaseID          SYSLIB-STORE-CT-3483
sl@0
   513
@SYMTestCaseDesc        The test is performed on media, where atomic "block write" operations are guaranteed.
sl@0
   514
						In a "file I/O error" simulation loop:
sl@0
   515
						 - Creates a CPermanentStoreFile object, stores there carefully selected set of test data,
sl@0
   516
						   such that a base TOC reference is split over the block boundary;
sl@0
   517
						 - Compacts the store;
sl@0
   518
						 - Commits;
sl@0
   519
@SYMTestPriority        High
sl@0
   520
@SYMTestActions         Creates a CPermanentStoreFile object, stores there carefully selected set of test data,
sl@0
   521
						such that a base TOC reference is split over the block boundary. When the store is compacted
sl@0
   522
						and the used media supports atomic "block write" operations, the STORE will update
sl@0
   523
						the base TOC reference, which requires a "file write" operation crossing the block boundary.
sl@0
   524
@SYMTestExpectedResults The test should not fail or panic.
sl@0
   525
@SYMREQ                 REQ0000
sl@0
   526
*/
sl@0
   527
void BaseTocReferenceTest()
sl@0
   528
	{
sl@0
   529
	TBaseTocReferenceTest baseTocReferenceTestObj;
sl@0
   530
	TocTest(baseTocReferenceTestObj);
sl@0
   531
	}
sl@0
   532
sl@0
   533
///////////////////////////////////////////////////////////////////////////////////////
sl@0
   534
sl@0
   535
void DoTests()
sl@0
   536
	{
sl@0
   537
	TheTest.Next(_L(" @SYMTestCaseID:SYSLIB-STORE-CT-3481 Base TOC entry test "));
sl@0
   538
	BaseTocEntryTest();
sl@0
   539
sl@0
   540
	TheTest.Next(_L(" @SYMTestCaseID:SYSLIB-STORE-CT-3482 Delta TOC entry test "));
sl@0
   541
	DeltaTocEntryTest();
sl@0
   542
sl@0
   543
	TheTest.Next(_L(" @SYMTestCaseID:SYSLIB-STORE-CT-3483 Base TOC reference test "));
sl@0
   544
	BaseTocReferenceTest();
sl@0
   545
	}
sl@0
   546
sl@0
   547
//The function returns true, if the file system guarantees atomic "block write" operations on aDriveNo.
sl@0
   548
TBool IsBlockAtomic(TInt aDriveNo)
sl@0
   549
	{
sl@0
   550
	__ASSERT_DEBUG(aDriveNo >= EDriveA && aDriveNo <= EDriveZ, User::Invariant());
sl@0
   551
sl@0
   552
	TVolumeIOParamInfo volInfo;
sl@0
   553
	TInt err = TheFs.VolumeIOParam(aDriveNo, volInfo);
sl@0
   554
	//If VolumeIOParam() succeeds, the media block size is >= 512 bytes and the media block size is power of two - report
sl@0
   555
	//that the media supports atomic "block write" operations.
sl@0
   556
	const TInt KDefaultMediaBlockSize = 512;
sl@0
   557
	return err == KErrNone && volInfo.iBlockSize >= KDefaultMediaBlockSize && (volInfo.iBlockSize & (volInfo.iBlockSize - 1)) == 0;
sl@0
   558
	}
sl@0
   559
sl@0
   560
//Returns the number of the drive on which atomic "block write" operations are supported.
sl@0
   561
TInt GetBlockAtomicDriveNo()
sl@0
   562
	{
sl@0
   563
	//Uncomment, if you want to do the test on drive C:
sl@0
   564
	//(but you will need also to uncomment 2 lines in CPermanentStoreCoord::FileQoSL())
sl@0
   565
	//return EDriveC;
sl@0
   566
	//
sl@0
   567
	for(TInt driveNo=EDriveA;driveNo<=EDriveZ;++driveNo)
sl@0
   568
		{
sl@0
   569
		TDriveInfo driveInfo;
sl@0
   570
		TInt err = TheFs.Drive(driveInfo, driveNo);
sl@0
   571
		if(err == KErrNone)
sl@0
   572
			{
sl@0
   573
			if(!(driveInfo.iDriveAtt & KDriveAttTransaction))
sl@0
   574
				{
sl@0
   575
#ifdef __X86GCC__
sl@0
   576
			TFSName thefsname;
sl@0
   577
			err = TheFs.FileSystemName (thefsname, driveNo);
sl@0
   578
sl@0
   579
			if ((err == KErrNone) && (!(thefsname.MatchF(_L("NTFS"))))) // X86GCC doesn't (yet) support write operations on NTFS partitions
sl@0
   580
#endif
sl@0
   581
				if(IsBlockAtomic(driveNo))
sl@0
   582
					{
sl@0
   583
					return driveNo;
sl@0
   584
					}
sl@0
   585
				}
sl@0
   586
			}
sl@0
   587
		}
sl@0
   588
	return -1;
sl@0
   589
	}
sl@0
   590
sl@0
   591
///////////////////////////////////////////////////////////////////////////////////////////////////////////////
sl@0
   592
// If you want to check that the atomic "block write" optimisation works and the code change is robust, then you
sl@0
   593
// have to:
sl@0
   594
// 1) Prove that if the the "file write" operations over a sector boundary are not done properly, the current test will crash
sl@0
   595
// 2) Prove that if the the "file write" operations over a sector boundary are done properly, the current test won't crash
sl@0
   596
// How this could be done:
sl@0
   597
// For (1) - Simulate that the file system for drive C: on the Emulator is transactional.
sl@0
   598
//    In order to do that:
sl@0
   599
//		- uncomment the "return EDriveC;" statement in the function above (GetBlockAtomicDriveNo());
sl@0
   600
//		- go to CPermanentStoreCoord::FileQoSL() function and uncomment the first two lines:
sl@0
   601
//			"iFileQos = ETransactional;"
sl@0
   602
//			"return iFileQos;"
sl@0
   603
//	  So, the result of your changes will be: STORE will try to optimize the "file write" operations on drive C:,
sl@0
   604
//	  because drive C: will be reported as a "transactional" drive. Since the drive C: does not really support "transactional"
sl@0
   605
//	  operations, the optimization will cause some failures in the current test, if the data block which has to be written,
sl@0
   606
//	  is split across a sector boundary. In order to make sure that the split will hapen - do step (2).
sl@0
   607
// 2) Declare SYSLIBS_TEST macro in the related variant.hrh file. This change will cause RFileBuf to split the "file write"
sl@0
   608
//	  operations in two separate operations, if the data block to be written has to be split over a sector boundary.
sl@0
   609
// After the changes, build STORE and run the current test. You must see that the test crashes.
sl@0
   610
//
sl@0
   611
// Then, for (2) (a) restore the original code in GetBlockAtomicDriveNo() and (b) CPermanentStoreCoord::FileQoSL() and rebuild STORE.
sl@0
   612
// Rerun the current test and the test must pass.
sl@0
   613
///////////////////////////////////////////////////////////////////////////////////////////////////////////////
sl@0
   614
sl@0
   615
TInt E32Main()
sl@0
   616
	{
sl@0
   617
	TheTest.Title();
sl@0
   618
sl@0
   619
	CTrapCleanup* tc = CTrapCleanup::New();
sl@0
   620
sl@0
   621
	__UHEAP_MARK;
sl@0
   622
sl@0
   623
	TInt err = TheFs.Connect();
sl@0
   624
	TheTest(err == KErrNone);
sl@0
   625
sl@0
   626
	TheTest.Start(_L("Check for \"block atomic write\" drives"));
sl@0
   627
	TInt driveNo = GetBlockAtomicDriveNo();
sl@0
   628
	if(driveNo != -1)
sl@0
   629
		{
sl@0
   630
		TheTest.Printf(_L("Test drive - %c:\r\n"), 'A' + driveNo);
sl@0
   631
		TheTest.Printf(_L("The test expects that STORE component is built with SYSLIBS_TEST macro defined\r\n"));
sl@0
   632
		CreateTestEnv(driveNo);
sl@0
   633
		DoTests();
sl@0
   634
		DeleteTestFiles();
sl@0
   635
		}
sl@0
   636
	else
sl@0
   637
		{
sl@0
   638
		TheTest.Printf(_L("!!! No \"block atomic write\" drive found!\r\n"));
sl@0
   639
		}
sl@0
   640
	TheFs.Close();
sl@0
   641
sl@0
   642
	__UHEAP_MARKEND;
sl@0
   643
sl@0
   644
	TheTest.End();
sl@0
   645
	TheTest.Close();
sl@0
   646
sl@0
   647
	delete tc;
sl@0
   648
sl@0
   649
	User::Heap().Check();
sl@0
   650
	return KErrNone;
sl@0
   651
	}