os/persistentdata/persistentstorage/store/TSTOR/T_BMPermFileStore.inl
author sl@SLION-WIN7.fritz.box
Fri, 15 Jun 2012 03:10:57 +0200
changeset 0 bde4ae8d615e
permissions -rw-r--r--
First public contribution.
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
// Demonstrates performance assesment of CPermanentFileStore
sl@0
    15
// The Store root stream serves as an example of persistent data structure
sl@0
    16
// 
sl@0
    17
//
sl@0
    18
sl@0
    19
// The file name, extension and path for the file store
sl@0
    20
_LIT(KPermFileStoreName,"Z:\\STOR-TST\\BMPermFileStores.dat");
sl@0
    21
extern TFileName ThePermStoreFilePath;
sl@0
    22
sl@0
    23
// Create a permanent file store and initialise its stream structure
sl@0
    24
LOCAL_C void doPrepareStoreL(const TDesC& aName);
sl@0
    25
sl@0
    26
// Use the store, preserving the stream structure
sl@0
    27
LOCAL_C void doPermFileStoreL(const TDesC& aName);
sl@0
    28
sl@0
    29
// Update the data in the store
sl@0
    30
LOCAL_C void doUpdateStoreL(CPersistentStore& aStore);
sl@0
    31
#define PFSTORERUNSIZE	1000
sl@0
    32
sl@0
    33
typedef TBuf<100> TItem;
sl@0
    34
sl@0
    35
// The main object's in-memory representation
sl@0
    36
class CItemArray : public CBase
sl@0
    37
	{
sl@0
    38
public:
sl@0
    39
	static TStreamId CreateL(CStreamStore& aStore);
sl@0
    40
	~CItemArray();
sl@0
    41
	static CItemArray* NewLC(CStreamStore& aStore,TStreamId anId);
sl@0
    42
	void RestoreL();
sl@0
    43
	void StoreL() const;
sl@0
    44
	void ExternalizeL(RWriteStream& aStream) const;
sl@0
    45
	//
sl@0
    46
	void AddItemL(const TItem& anItem);
sl@0
    47
	void RemoveItemL(TInt anIndex);
sl@0
    48
	TInt Count() const;
sl@0
    49
	void GetItemL(TItem& anItem,TInt anIndex) const;
sl@0
    50
protected:
sl@0
    51
	CItemArray(CStreamStore& aStore);
sl@0
    52
	void ConstructL();
sl@0
    53
	void InternalizeL(RReadStream& aStream);
sl@0
    54
private:
sl@0
    55
	CStreamStore& iStore;
sl@0
    56
	TStreamId iMyId;
sl@0
    57
	CArrayFixFlat<TStreamId>* iArray;
sl@0
    58
	CArrayFixFlat<TElement>* nArray;
sl@0
    59
	};
sl@0
    60
sl@0
    61
/**
sl@0
    62
@SYMTestCaseID          SYSLIB-STORE-PT-1371
sl@0
    63
@SYMTestCaseDesc	CR MRLM-6A9DF7. Tests the impact of RFileBuf cache size on CPermanentFileStore performance. RFileBuf cache size must be set at a build time in EStor_Template.mmh file, DEFAULT_FILE_BUF_SIZE macro definition.
sl@0
    64
@SYMTestPriority 	High
sl@0
    65
@SYMTestActions  	The test creates a set of test objects, externalizes them PFSTORERUNSIZE times and prints the execution time. Then internalizes them PFSTORERUNSIZE times and prints the execution time. PFSTORERUNSIZE is a constant, defined in the source file.
sl@0
    66
@SYMTestExpectedResults Test must show a performance improvement
sl@0
    67
@SYMPREQ                PREQ1132
sl@0
    68
@SYMREQ			REQ4883
sl@0
    69
*/
sl@0
    70
LOCAL_C void doPermanentFileStoreL()
sl@0
    71
    {
sl@0
    72
    TDriveUnit drive(static_cast<TUint>(RFs::GetSystemDrive()));	
sl@0
    73
	TParse permFileStoreName;
sl@0
    74
	permFileStoreName.Set(drive.Name(), &KPermFileStoreName, NULL);
sl@0
    75
	ThePermStoreFilePath.Copy(permFileStoreName.FullName());
sl@0
    76
    
sl@0
    77
	TheFs.MkDirAll(permFileStoreName.DriveAndPath());
sl@0
    78
	doPrepareStoreL(ThePermStoreFilePath);
sl@0
    79
	
sl@0
    80
	TTime start;
sl@0
    81
	start.HomeTime();
sl@0
    82
	for(int i=0; i<PFSTORERUNSIZE; i++)
sl@0
    83
		{
sl@0
    84
		doPermFileStoreL(ThePermStoreFilePath);
sl@0
    85
		}
sl@0
    86
	TTime end;
sl@0
    87
	end.HomeTime();
sl@0
    88
	TTimeIntervalMicroSeconds us = end.MicroSecondsFrom(start);
sl@0
    89
	TheTest.Printf(_L("CPermanentFileStore, %d 'write' tests. Time=%ld ms\r\n"), PFSTORERUNSIZE, us.Int64() / 1000);
sl@0
    90
	}
sl@0
    91
sl@0
    92
LOCAL_C void doPrepareStoreL(const TDesC& aName)
sl@0
    93
	{
sl@0
    94
				// construct file store object - the file to contain the
sl@0
    95
				// the store replaces any existing file of the same name.
sl@0
    96
	CFileStore* store = CPermanentFileStore::ReplaceLC(TheFs,aName,EFileRead|EFileWrite);
sl@0
    97
				// Easy way to set the layout type
sl@0
    98
    store->SetTypeL(store->Layout());
sl@0
    99
		
sl@0
   100
				// create the required stream for CItemArray
sl@0
   101
	TStreamId id=CItemArray::CreateL(*store);
sl@0
   102
	store->SetRootL(id);
sl@0
   103
	store->CommitL();
sl@0
   104
	CleanupStack::PopAndDestroy();
sl@0
   105
	}
sl@0
   106
sl@0
   107
LOCAL_C void doPermFileStoreL(const TDesC& aName)
sl@0
   108
	{
sl@0
   109
				// construct file store object - specifying the file
sl@0
   110
				// containing the store.
sl@0
   111
				// Do not need to specify the file store type, this is
sl@0
   112
				// specified by the file itself
sl@0
   113
	CFileStore* store = CFileStore::OpenL(TheFs,aName,EFileRead|EFileWrite);
sl@0
   114
sl@0
   115
				// The standard form for using permanent file stores:
sl@0
   116
				// 1. The store object is not owned by the updating code
sl@0
   117
				// 2  Failure at any point during update, including the
sl@0
   118
				//    final commit, should result in Revert() being called
sl@0
   119
				//    on the store (before destruction).
sl@0
   120
	_LIT(KTxtErrorOccurred,"\n** Error %d occured during store update");
sl@0
   121
sl@0
   122
	TRAPD(error,doUpdateStoreL(*store));
sl@0
   123
	if (error!=KErrNone)
sl@0
   124
		{
sl@0
   125
		store->Revert();
sl@0
   126
		TheTest.Printf(KTxtErrorOccurred);
sl@0
   127
		}
sl@0
   128
sl@0
   129
	// the store is not on the cleanup stack
sl@0
   130
	delete store;
sl@0
   131
	}
sl@0
   132
sl@0
   133
sl@0
   134
LOCAL_C void doUpdateStoreL(CPersistentStore& aStore)
sl@0
   135
	{
sl@0
   136
				// get the root stream into memory
sl@0
   137
	CItemArray* array=CItemArray::NewLC(aStore,aStore.Root());
sl@0
   138
sl@0
   139
				// Add some items
sl@0
   140
	_LIT(KTxtHello,"hello");
sl@0
   141
	_LIT(KTxtWorld," world!");
sl@0
   142
sl@0
   143
	TItem item;
sl@0
   144
	item = KTxtHello;
sl@0
   145
	array->AddItemL(item);
sl@0
   146
	item = KTxtWorld;
sl@0
   147
	array->AddItemL(item);
sl@0
   148
				// Re-write the root stream with new data
sl@0
   149
	array->StoreL();
sl@0
   150
	aStore.CommitL();
sl@0
   151
sl@0
   152
				// remove an item
sl@0
   153
	array->RemoveItemL(1);		// " world!"
sl@0
   154
				// Re-write the root stream with new data
sl@0
   155
	array->StoreL();
sl@0
   156
	aStore.CommitL();
sl@0
   157
				// Add an item
sl@0
   158
	_LIT(KTxtCapWorld," WORLD!");
sl@0
   159
	item= KTxtCapWorld;
sl@0
   160
	array->AddItemL(item);
sl@0
   161
				// Re-write the root stream with new data
sl@0
   162
	array->StoreL();
sl@0
   163
				// Discard all changes since last store commit
sl@0
   164
	aStore.Revert();
sl@0
   165
				// array and aStore are not in snych after revert...
sl@0
   166
				// restore in-memory version to match store version
sl@0
   167
	array->RestoreL();
sl@0
   168
					// Add the item again
sl@0
   169
	array->AddItemL(item);
sl@0
   170
				// Re-write the root stream with new data
sl@0
   171
	array->StoreL();
sl@0
   172
	aStore.CommitL();
sl@0
   173
	CleanupStack::PopAndDestroy(); // array
sl@0
   174
	}
sl@0
   175
sl@0
   176
//***************************************************************
sl@0
   177
CItemArray::~CItemArray()
sl@0
   178
	{
sl@0
   179
	delete iArray;
sl@0
   180
	delete nArray;
sl@0
   181
	}
sl@0
   182
sl@0
   183
CItemArray::CItemArray(CStreamStore& aStore)
sl@0
   184
	: iStore(aStore)
sl@0
   185
	{}
sl@0
   186
sl@0
   187
TStreamId CItemArray::CreateL(CStreamStore& aStore)
sl@0
   188
// create the stream representation of the class
sl@0
   189
	{
sl@0
   190
				// use a temporary CItemArray
sl@0
   191
	CItemArray* self=new(ELeave) CItemArray(aStore);
sl@0
   192
	CleanupStack::PushL(self);
sl@0
   193
				// construct object
sl@0
   194
	self->ConstructL();
sl@0
   195
				// create new stream
sl@0
   196
	RStoreWriteStream outstream;
sl@0
   197
	TStreamId id=outstream.CreateLC(aStore);
sl@0
   198
				// write  external rep
sl@0
   199
	self->ExternalizeL(outstream);
sl@0
   200
				// commit stream
sl@0
   201
	outstream.CommitL();
sl@0
   202
				// cleanup stream and temporary self
sl@0
   203
	CleanupStack::PopAndDestroy(2);
sl@0
   204
	return id;
sl@0
   205
	}
sl@0
   206
sl@0
   207
CItemArray* CItemArray::NewLC(CStreamStore& aStore,TStreamId anId)
sl@0
   208
// construct a CItemArray from persistent storage
sl@0
   209
	{
sl@0
   210
	CItemArray* self=new(ELeave) CItemArray(aStore);
sl@0
   211
	CleanupStack::PushL(self);
sl@0
   212
				// construct object
sl@0
   213
	self->ConstructL();
sl@0
   214
				// set the stream id for StoreL/RestoreL
sl@0
   215
	self->iMyId=anId;
sl@0
   216
				// restore the internal rep.
sl@0
   217
	self->RestoreL();
sl@0
   218
	return self;
sl@0
   219
	}
sl@0
   220
sl@0
   221
void CItemArray::StoreL() const
sl@0
   222
// replace external rep. with internal one
sl@0
   223
	{
sl@0
   224
	RStoreWriteStream outstream;
sl@0
   225
	outstream.ReplaceLC(iStore,iMyId);
sl@0
   226
	ExternalizeL(outstream);
sl@0
   227
	outstream.CommitL();
sl@0
   228
	CleanupStack::PopAndDestroy();
sl@0
   229
	}
sl@0
   230
sl@0
   231
void CItemArray::RestoreL()
sl@0
   232
// replace internal rep with external one
sl@0
   233
	{
sl@0
   234
	iArray->Reset();
sl@0
   235
	nArray->Reset();
sl@0
   236
	RStoreReadStream instream;
sl@0
   237
	instream.OpenLC(iStore,iMyId);
sl@0
   238
	InternalizeL(instream);
sl@0
   239
	CleanupStack::PopAndDestroy();
sl@0
   240
	}
sl@0
   241
sl@0
   242
void CItemArray::AddItemL(const TItem& anItem)
sl@0
   243
// add item to the collection
sl@0
   244
	{
sl@0
   245
				// write external rep of item
sl@0
   246
	RStoreWriteStream outstream;
sl@0
   247
	TStreamId id=outstream.CreateLC(iStore);
sl@0
   248
	outstream<<anItem;
sl@0
   249
	outstream.CommitL();
sl@0
   250
	CleanupStack::PopAndDestroy();
sl@0
   251
				// add new stream id to the internal array
sl@0
   252
	iArray->AppendL(id);
sl@0
   253
	}
sl@0
   254
sl@0
   255
void CItemArray::RemoveItemL(TInt anIndex)
sl@0
   256
// remove an item from the collection
sl@0
   257
	{
sl@0
   258
				// remove the stream from the store
sl@0
   259
	iStore.DeleteL((*iArray)[anIndex]);
sl@0
   260
				// remove the entry from the internal array
sl@0
   261
	iArray->Delete(anIndex);
sl@0
   262
	}
sl@0
   263
sl@0
   264
TInt CItemArray::Count() const
sl@0
   265
	{
sl@0
   266
	return iArray->Count();
sl@0
   267
	}
sl@0
   268
sl@0
   269
void CItemArray::GetItemL(TItem& anItem,TInt anIndex) const
sl@0
   270
	{
sl@0
   271
				// retrieve an item from the store
sl@0
   272
	RStoreReadStream instream;
sl@0
   273
	instream.OpenLC(iStore,(*iArray)[anIndex]);
sl@0
   274
	instream>>anItem;
sl@0
   275
	CleanupStack::PopAndDestroy();
sl@0
   276
	}
sl@0
   277
sl@0
   278
void CItemArray::ConstructL()
sl@0
   279
	{
sl@0
   280
	iArray=new(ELeave) CArrayFixFlat<TStreamId>(8);
sl@0
   281
	nArray = new (ELeave) CArrayFixFlat<TElement>(PFSTORERUNSIZE);
sl@0
   282
	TElement theElement;
sl@0
   283
	_LIT(KFormatTxt,"BenchMarkingPermanentFileStore%4u");
sl@0
   284
	TBuf<256> str(KFormatTxt);
sl@0
   285
sl@0
   286
	for (TInt index = 0; index < PFSTORERUNSIZE; index++)
sl@0
   287
		{
sl@0
   288
		theElement.iData.Format(KFormatTxt,index);
sl@0
   289
		nArray->AppendL(theElement);
sl@0
   290
		}
sl@0
   291
	}
sl@0
   292
sl@0
   293
void CItemArray::ExternalizeL(RWriteStream& aStream) const
sl@0
   294
	{
sl@0
   295
				// stream out the array
sl@0
   296
	aStream<<*iArray;
sl@0
   297
	TInt count = nArray->Count();
sl@0
   298
	TElement theElement;
sl@0
   299
	for (TInt index = 0; index < count; index++)
sl@0
   300
		{
sl@0
   301
		theElement =((*nArray)[index]);
sl@0
   302
		aStream.WriteL(theElement.iData, 34);
sl@0
   303
		}
sl@0
   304
	}
sl@0
   305
sl@0
   306
void CItemArray::InternalizeL(RReadStream& aStream)
sl@0
   307
	{
sl@0
   308
				// stream in the array
sl@0
   309
	aStream>>*iArray;
sl@0
   310
	
sl@0
   311
	TElement theElement;
sl@0
   312
	TInt count = nArray->Count();
sl@0
   313
	nArray->Reset();
sl@0
   314
	for (TInt index = 0; index < count; index++)
sl@0
   315
		{
sl@0
   316
		aStream.ReadL(theElement.iData, 34);
sl@0
   317
		nArray->AppendL(theElement);
sl@0
   318
		}
sl@0
   319
	}