Update contrib.
1 // Copyright (c) 1998-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.
20 EXPORT_C void TStorePagePoolToken::ExternalizeL(RWriteStream& aStream) const
21 /** Externalises a TStorePagePoolToken object to a stream.
23 @param aStream Stream to which the object should be externalised */
29 EXPORT_C void TStorePagePoolToken::InternalizeL(RReadStream& aStream)
30 /** Internalises a TStorePagePoolToken object from a stream.
32 @param aStream Stream from which the object should be internalised */
38 // Class RStorePagePool
40 EXPORT_C RStorePagePool::RStorePagePool()
42 /** Default constructor. */
45 EXPORT_C RStorePagePool::RStorePagePool(CPageCache& aCache)
46 : TCachePagePool(aCache),iStore(NULL)
47 /** Constructor with a page cache for the pool.
49 @param aCache Page cache for the pool */
52 EXPORT_C RStorePagePool::RStorePagePool(CStreamStore& aStore)
53 /** Constructor with a stream store to use for the pool.
55 @param aStore Stream store to use for the pool */
60 EXPORT_C RStorePagePool::RStorePagePool(CStreamStore& aStore,const TStorePagePoolToken& aToken)
61 /** Constructor with a stream store and settings to use for the pool.
63 @param aStore Stream store to use for the pool
64 @param aToken Stream store pool settings */
69 EXPORT_C void RStorePagePool::Create(CStreamStore& aStore)
70 /** Creates a new pool.
72 @param aStore Stream store to use for the pool */
80 EXPORT_C void RStorePagePool::Open(CStreamStore& aStore,const TStorePagePoolToken& aToken)
81 /** Opens an existing pool.
83 @param aStore Stream store for the pool
84 @param aToken Pool settings */
92 EXPORT_C TStorePagePoolToken RStorePagePool::Token() const
93 /** Gets an object that encapsulates the page pool settings.
95 That object can then be used to externalise the settings.
97 @return Encapsulates the page pool settings */
99 return TStorePagePoolToken(iHead,iAvail);
102 EXPORT_C void RStorePagePool::Close()
103 /** Flushes and purges the page cache and stops using the stream store. */
105 TCachePagePool::Flush();
109 EXPORT_C TBool RStorePagePool::ReclaimL()
110 /** Deletes the first stream that stores reclaimable pages in the store.
112 @return True if there are remaining streams (pages), false if there are no
115 __ASSERT_DEBUG(iStore!=NULL,Panic(EPageNotOpen));
116 __ASSERT_ALWAYS(iAvail==KNullPageRef,Panic(EPageReclaimAvailable));
117 if (iHead!=KNullStreamId)
119 RStoreReadStream out;
120 out.OpenLC(*iStore,iHead);
123 CleanupStack::PopAndDestroy();
124 iStore->DeleteL(iHead);
127 if (next!=KNullStreamId)
133 EXPORT_C void RStorePagePool::ReclaimAllL()
134 /** Deletes all streams in the store. */
140 EXPORT_C TPageRef RStorePagePool::ExtendL(const TAny* aPage,TPageReclamation aReclamation)
142 // Create a new page.
145 return StorePagePool::ExtendL(*this,aPage,aReclamation);
148 EXPORT_C void RStorePagePool::WriteL(TPageRef aRef,const TAny* aPage,TPageChange aChange)
150 // Write a page to the store.
153 StorePagePool::WriteL(*this,aRef,aPage,aChange);
156 EXPORT_C void RStorePagePool::ReadL(TPageRef aRef,TAny* aPage)
158 // Read from the store into a page.
161 StorePagePool::ReadL(*this,aRef,aPage);
164 EXPORT_C void RStorePagePool::DoDeleteL(TPageRef aRef)
166 // Delete from store.
169 StorePagePool::DeleteL(*this,aRef);
172 // Class RSecureStorePagePool
176 EXPORT_C RSecureStorePagePool::RSecureStorePagePool(const CPBEncryptSet& aKey)
180 EXPORT_C RSecureStorePagePool::RSecureStorePagePool(CPageCache& aCache, const CPBEncryptSet& aKey)
181 : RStorePagePool(aCache),
185 /** Adds a new page to the pool.
187 @param aPage Data for the page
188 @param aReclamation Flags that define how allocated pages can be reclaimed
189 @return Reference to newly created page */
190 EXPORT_C TPageRef RSecureStorePagePool::ExtendL(const TAny* aPage,TPageReclamation aReclamation)
192 return StorePagePool::ExtendL(*this,aPage,aReclamation);
195 /** Writes data to a page in the pool.
197 @param aRef Reference to the page to which to write
198 @param aPage Data to write
199 @param aChange Flags that define how a page should be treated when it is unlocked
201 EXPORT_C void RSecureStorePagePool::WriteL(TPageRef aRef,const TAny* aPage,TPageChange aChange)
203 StorePagePool::WriteL(*this,aRef,aPage,aChange);
206 /** Reads a specified page from the pool.
208 @param aRef Reference to page to read
209 @param aPage On return, the page data */
210 EXPORT_C void RSecureStorePagePool::ReadL(TPageRef aRef,TAny* aPage)
212 StorePagePool::ReadL(*this,aRef,aPage);
215 /** Deletes a specified page.
217 @param aRef Reference to page to delete */
218 EXPORT_C void RSecureStorePagePool::DoDeleteL(TPageRef aRef)
220 StorePagePool::DeleteL(*this,aRef);
223 // Class StorePagePool
224 // the implementation of the clever bits to share code between the secure and plain sotre page pools
226 const TInt KMaxPageIndex=15;
228 inline TInt PageSize(const CPBEncryptionBase* aKey)
230 return aKey ? aKey->MaxCiphertextLength(KPoolPageSize) : KPoolPageSize;
233 inline TPageRef PageRef(TStreamId anId,TInt anIndex=0)
235 __ASSERT_DEBUG(anIndex>=0&&anIndex<=KMaxPageIndex,User::Invariant());
236 return TPageRef((anId.Value()<<4)+anIndex);
238 inline TInt PageIndex(TPageRef aRef)
239 {return aRef.Value()&0xf;}
240 inline TStreamId StreamId(TPageRef aRef)
241 {return TStreamId(aRef.Value()>>4);}
242 inline TStreamPos PagePos(TInt anIndex,TInt aPageSize)
243 {return TStreamPos(sizeof(TStreamId)+(anIndex-1)*aPageSize);}
245 void StorePagePool::PadL(RWriteStream& aStream,TInt aLength)
250 TUint8 zero[KPoolPageSize];
251 Mem::FillZ(zero,KPoolPageSize);
252 while ((aLength-=KPoolPageSize)>0)
253 aStream.WriteL(zero,KPoolPageSize);
254 aStream.WriteL(zero,aLength+KPoolPageSize);
258 // Encrypt the page to the stream, padding out to full page size
259 void StorePagePool::EncryptL(RWriteStream& aStream,const TAny* aPage,const CPBEncryptionBase& aKey)
261 REncryptStream encrypt;
262 encrypt.OpenLC(aStream,aKey);
263 encrypt.WriteL((const TUint8*)aPage,KPoolPageSize);
265 CleanupStack::PopAndDestroy(&encrypt);
269 // Encrypt the page to the stream, padding out to full page size
271 void StorePagePool::EncryptNewL(RWriteStream& aStream,const TAny* aPage,const CPBEncryptionBase& aKey)
273 TStreamPos pos=aStream.Sink()->TellL(MStreamBuf::EWrite); // remember the start of the data
274 EncryptL(aStream,aPage,aKey);
275 TInt pad=PageSize(&aKey)-(aStream.Sink()->TellL(MStreamBuf::EWrite)-pos); // bytes written
276 __ASSERT_DEBUG(pad>=0,Panic(EPageCipherTextOverrun));
282 // Encrypt the page to the stream, padding out to full page size
284 void StorePagePool::DecryptL(RReadStream& aStream,TAny* aPage,const CPBEncryptionBase& aKey)
286 RDecryptStream decrypt;
287 decrypt.OpenLC(aStream,aKey);
288 decrypt.ReadL((TUint8*)aPage,KPoolPageSize);
289 CleanupStack::PopAndDestroy();
292 void StorePagePool::SeekL(MStreamBuf* aBuf,TInt aMark,TPageRef aRef,const CPBEncryptionBase* aKey)
294 TInt ix=PageIndex(aRef);
296 aBuf->SeekL(aMark,PagePos(ix,PageSize(aKey)));
299 TPageRef StorePagePool::ExtendL(RStorePagePool& aPool,const TAny* aPage,TPageReclamation aReclamation,const CPBEncryptionBase* aKey)
301 __ASSERT_DEBUG(aPool.iStore!=NULL,Panic(EPageNotOpen));
303 RStoreWriteStream out;
304 if (aReclamation==EPageDeleteOnly)
305 { // non-reclaimable page
306 page=PageRef(out.CreateLC(*aPool.iStore));
308 else if (aPool.iAvail!=KNullPageRef)
311 out.OpenLC(*aPool.iStore,StreamId(page));
312 MStreamBuf* buf=out.Sink();
313 SeekL(buf,MStreamBuf::ERead|MStreamBuf::EWrite,page,aKey);
319 { // allocate new block, and create free list
320 TStreamId id=out.CreateLC(*aPool.iStore);
321 out<<aPool.iHead; // block list
322 const TInt pad=PageSize(aKey)-sizeof(TPageRef);
323 page=KNullPageRef; // free-list terminator
324 for (TInt index=0;++index<KMaxPageIndex;)
328 page=PageRef(id,index);
333 page=PageRef(id,KMaxPageIndex); // page is written at end
336 EncryptNewL(out,aPage,*aKey);
338 out.WriteL((const TUint8*)aPage,KPoolPageSize);
340 CleanupStack::PopAndDestroy();
344 void StorePagePool::WriteL(RStorePagePool& aPool,TPageRef aRef,const TAny* aPage,TPageChange aChange,const CPBEncryptionBase* aKey)
346 __ASSERT_DEBUG(aChange>=EPageDirty,Panic(EPageChangeInvalid));
347 __ASSERT_DEBUG(aPool.iStore!=NULL,Panic(EPageNotOpen));
348 RStoreWriteStream out;
349 if (aChange==EPageUpdate)
351 __ASSERT_DEBUG(PageIndex(aRef)==0,Panic(EPageChangeInvalid));
352 out.ReplaceLC(*aPool.iStore,StreamId(aRef));
354 EncryptNewL(out,aPage,*aKey);
356 out.WriteL((const TUint8*)aPage,KPoolPageSize);
359 { // re-write the page, no padding req'd
360 out.OpenLC(*aPool.iStore,StreamId(aRef));
361 SeekL(out.Sink(),MStreamBuf::EWrite,aRef,aKey);
363 EncryptL(out,aPage,*aKey);
365 out.WriteL((const TUint8*)aPage,KPoolPageSize);
368 CleanupStack::PopAndDestroy();
371 void StorePagePool::ReadL(RStorePagePool& aPool,TPageRef aRef,TAny* aPage,const CPBEncryptionBase* aKey)
373 __ASSERT_DEBUG(aPool.iStore!=NULL,Panic(EPageNotOpen));
375 in.OpenLC(*aPool.iStore,StreamId(aRef));
376 SeekL(in.Source(),MStreamBuf::ERead,aRef,aKey);
378 DecryptL(in,aPage,*aKey);
380 in.ReadL((TUint8*)aPage,KPoolPageSize);
381 CleanupStack::PopAndDestroy(); // in
384 void StorePagePool::DeleteL(RStorePagePool& aPool,TPageRef aRef,const CPBEncryptionBase* aKey)
386 aPool.CacheDeleteL(aRef);
387 __ASSERT_DEBUG(aPool.iStore!=NULL,Panic(EPageNotOpen));
388 if (PageIndex(aRef)==0)
390 aPool.iStore->DeleteL(StreamId(aRef));
394 RStoreWriteStream out;
395 out.OpenLC(*aPool.iStore,StreamId(aRef));
396 SeekL(out.Sink(),MStreamBuf::EWrite,aRef,aKey);
399 CleanupStack::PopAndDestroy();