First public contribution.
1 // Copyright (c) 2008-2009 Nokia Corporation and/or its subsidiary(-ies).
2 // All rights reserved.
3 // This component and the accompanying materials are made available
4 // under the terms of "Eclipse Public License v1.0"
5 // which accompanies this distribution, and is available
6 // at the URL "http://www.eclipse.org/legal/epl-v10.html".
8 // Initial Contributors:
9 // Nokia Corporation - initial contribution.
14 // Demonstrates performance assesment of CPermanentFileStore
15 // The Store root stream serves as an example of persistent data structure
19 // The file name, extension and path for the file store
20 _LIT(KPermFileStoreName,"Z:\\STOR-TST\\BMPermFileStores.dat");
21 extern TFileName ThePermStoreFilePath;
23 // Create a permanent file store and initialise its stream structure
24 LOCAL_C void doPrepareStoreL(const TDesC& aName);
26 // Use the store, preserving the stream structure
27 LOCAL_C void doPermFileStoreL(const TDesC& aName);
29 // Update the data in the store
30 LOCAL_C void doUpdateStoreL(CPersistentStore& aStore);
31 #define PFSTORERUNSIZE 1000
33 typedef TBuf<100> TItem;
35 // The main object's in-memory representation
36 class CItemArray : public CBase
39 static TStreamId CreateL(CStreamStore& aStore);
41 static CItemArray* NewLC(CStreamStore& aStore,TStreamId anId);
44 void ExternalizeL(RWriteStream& aStream) const;
46 void AddItemL(const TItem& anItem);
47 void RemoveItemL(TInt anIndex);
49 void GetItemL(TItem& anItem,TInt anIndex) const;
51 CItemArray(CStreamStore& aStore);
53 void InternalizeL(RReadStream& aStream);
57 CArrayFixFlat<TStreamId>* iArray;
58 CArrayFixFlat<TElement>* nArray;
62 @SYMTestCaseID SYSLIB-STORE-PT-1371
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.
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.
66 @SYMTestExpectedResults Test must show a performance improvement
70 LOCAL_C void doPermanentFileStoreL()
72 TDriveUnit drive(static_cast<TUint>(RFs::GetSystemDrive()));
73 TParse permFileStoreName;
74 permFileStoreName.Set(drive.Name(), &KPermFileStoreName, NULL);
75 ThePermStoreFilePath.Copy(permFileStoreName.FullName());
77 TheFs.MkDirAll(permFileStoreName.DriveAndPath());
78 doPrepareStoreL(ThePermStoreFilePath);
82 for(int i=0; i<PFSTORERUNSIZE; i++)
84 doPermFileStoreL(ThePermStoreFilePath);
88 TTimeIntervalMicroSeconds us = end.MicroSecondsFrom(start);
89 TheTest.Printf(_L("CPermanentFileStore, %d 'write' tests. Time=%ld ms\r\n"), PFSTORERUNSIZE, us.Int64() / 1000);
92 LOCAL_C void doPrepareStoreL(const TDesC& aName)
94 // construct file store object - the file to contain the
95 // the store replaces any existing file of the same name.
96 CFileStore* store = CPermanentFileStore::ReplaceLC(TheFs,aName,EFileRead|EFileWrite);
97 // Easy way to set the layout type
98 store->SetTypeL(store->Layout());
100 // create the required stream for CItemArray
101 TStreamId id=CItemArray::CreateL(*store);
104 CleanupStack::PopAndDestroy();
107 LOCAL_C void doPermFileStoreL(const TDesC& aName)
109 // construct file store object - specifying the file
110 // containing the store.
111 // Do not need to specify the file store type, this is
112 // specified by the file itself
113 CFileStore* store = CFileStore::OpenL(TheFs,aName,EFileRead|EFileWrite);
115 // The standard form for using permanent file stores:
116 // 1. The store object is not owned by the updating code
117 // 2 Failure at any point during update, including the
118 // final commit, should result in Revert() being called
119 // on the store (before destruction).
120 _LIT(KTxtErrorOccurred,"\n** Error %d occured during store update");
122 TRAPD(error,doUpdateStoreL(*store));
126 TheTest.Printf(KTxtErrorOccurred);
129 // the store is not on the cleanup stack
134 LOCAL_C void doUpdateStoreL(CPersistentStore& aStore)
136 // get the root stream into memory
137 CItemArray* array=CItemArray::NewLC(aStore,aStore.Root());
140 _LIT(KTxtHello,"hello");
141 _LIT(KTxtWorld," world!");
145 array->AddItemL(item);
147 array->AddItemL(item);
148 // Re-write the root stream with new data
153 array->RemoveItemL(1); // " world!"
154 // Re-write the root stream with new data
158 _LIT(KTxtCapWorld," WORLD!");
160 array->AddItemL(item);
161 // Re-write the root stream with new data
163 // Discard all changes since last store commit
165 // array and aStore are not in snych after revert...
166 // restore in-memory version to match store version
168 // Add the item again
169 array->AddItemL(item);
170 // Re-write the root stream with new data
173 CleanupStack::PopAndDestroy(); // array
176 //***************************************************************
177 CItemArray::~CItemArray()
183 CItemArray::CItemArray(CStreamStore& aStore)
187 TStreamId CItemArray::CreateL(CStreamStore& aStore)
188 // create the stream representation of the class
190 // use a temporary CItemArray
191 CItemArray* self=new(ELeave) CItemArray(aStore);
192 CleanupStack::PushL(self);
196 RStoreWriteStream outstream;
197 TStreamId id=outstream.CreateLC(aStore);
198 // write external rep
199 self->ExternalizeL(outstream);
202 // cleanup stream and temporary self
203 CleanupStack::PopAndDestroy(2);
207 CItemArray* CItemArray::NewLC(CStreamStore& aStore,TStreamId anId)
208 // construct a CItemArray from persistent storage
210 CItemArray* self=new(ELeave) CItemArray(aStore);
211 CleanupStack::PushL(self);
214 // set the stream id for StoreL/RestoreL
216 // restore the internal rep.
221 void CItemArray::StoreL() const
222 // replace external rep. with internal one
224 RStoreWriteStream outstream;
225 outstream.ReplaceLC(iStore,iMyId);
226 ExternalizeL(outstream);
228 CleanupStack::PopAndDestroy();
231 void CItemArray::RestoreL()
232 // replace internal rep with external one
236 RStoreReadStream instream;
237 instream.OpenLC(iStore,iMyId);
238 InternalizeL(instream);
239 CleanupStack::PopAndDestroy();
242 void CItemArray::AddItemL(const TItem& anItem)
243 // add item to the collection
245 // write external rep of item
246 RStoreWriteStream outstream;
247 TStreamId id=outstream.CreateLC(iStore);
250 CleanupStack::PopAndDestroy();
251 // add new stream id to the internal array
255 void CItemArray::RemoveItemL(TInt anIndex)
256 // remove an item from the collection
258 // remove the stream from the store
259 iStore.DeleteL((*iArray)[anIndex]);
260 // remove the entry from the internal array
261 iArray->Delete(anIndex);
264 TInt CItemArray::Count() const
266 return iArray->Count();
269 void CItemArray::GetItemL(TItem& anItem,TInt anIndex) const
271 // retrieve an item from the store
272 RStoreReadStream instream;
273 instream.OpenLC(iStore,(*iArray)[anIndex]);
275 CleanupStack::PopAndDestroy();
278 void CItemArray::ConstructL()
280 iArray=new(ELeave) CArrayFixFlat<TStreamId>(8);
281 nArray = new (ELeave) CArrayFixFlat<TElement>(PFSTORERUNSIZE);
283 _LIT(KFormatTxt,"BenchMarkingPermanentFileStore%4u");
284 TBuf<256> str(KFormatTxt);
286 for (TInt index = 0; index < PFSTORERUNSIZE; index++)
288 theElement.iData.Format(KFormatTxt,index);
289 nArray->AppendL(theElement);
293 void CItemArray::ExternalizeL(RWriteStream& aStream) const
295 // stream out the array
297 TInt count = nArray->Count();
299 for (TInt index = 0; index < count; index++)
301 theElement =((*nArray)[index]);
302 aStream.WriteL(theElement.iData, 34);
306 void CItemArray::InternalizeL(RReadStream& aStream)
308 // stream in the array
312 TInt count = nArray->Count();
314 for (TInt index = 0; index < count; index++)
316 aStream.ReadL(theElement.iData, 34);
317 nArray->AppendL(theElement);