os/persistentdata/persistentstorage/store/UPAGE/UP_STOR.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 "UP_STD.H"
sl@0
    17
sl@0
    18
#include <pbe.h>
sl@0
    19
sl@0
    20
EXPORT_C void TStorePagePoolToken::ExternalizeL(RWriteStream& aStream) const
sl@0
    21
/** Externalises a TStorePagePoolToken object to a stream.
sl@0
    22
sl@0
    23
@param aStream Stream to which the object should be externalised */
sl@0
    24
	{
sl@0
    25
	aStream<<iHead;
sl@0
    26
	aStream<<iAvail;
sl@0
    27
	}
sl@0
    28
sl@0
    29
EXPORT_C void TStorePagePoolToken::InternalizeL(RReadStream& aStream)
sl@0
    30
/** Internalises a TStorePagePoolToken object from a stream.
sl@0
    31
sl@0
    32
@param aStream Stream from which the object should be internalised */
sl@0
    33
	{
sl@0
    34
	aStream>>iHead;
sl@0
    35
	aStream>>iAvail;
sl@0
    36
	}
sl@0
    37
sl@0
    38
// Class RStorePagePool
sl@0
    39
sl@0
    40
EXPORT_C RStorePagePool::RStorePagePool()
sl@0
    41
	: iStore(NULL)
sl@0
    42
/** Default constructor. */
sl@0
    43
	{}
sl@0
    44
sl@0
    45
EXPORT_C RStorePagePool::RStorePagePool(CPageCache& aCache)
sl@0
    46
	: TCachePagePool(aCache),iStore(NULL)
sl@0
    47
/** Constructor with a page cache for the pool.
sl@0
    48
sl@0
    49
@param aCache Page cache for the pool */
sl@0
    50
	{}
sl@0
    51
sl@0
    52
EXPORT_C RStorePagePool::RStorePagePool(CStreamStore& aStore)
sl@0
    53
/** Constructor with a stream store to use for the pool.
sl@0
    54
sl@0
    55
@param aStore Stream store to use for the pool */
sl@0
    56
	{
sl@0
    57
	Create(aStore);
sl@0
    58
	}
sl@0
    59
sl@0
    60
EXPORT_C RStorePagePool::RStorePagePool(CStreamStore& aStore,const TStorePagePoolToken& aToken)
sl@0
    61
/** Constructor with a stream store and settings to use for the pool.
sl@0
    62
sl@0
    63
@param aStore Stream store to use for the pool
sl@0
    64
@param aToken Stream store pool settings */
sl@0
    65
	{
sl@0
    66
	Open(aStore,aToken);
sl@0
    67
	}
sl@0
    68
sl@0
    69
EXPORT_C void RStorePagePool::Create(CStreamStore& aStore)
sl@0
    70
/** Creates a new pool.
sl@0
    71
sl@0
    72
@param aStore Stream store to use for the pool */
sl@0
    73
	{
sl@0
    74
	iStore=&aStore;
sl@0
    75
	iHead=KNullStreamId;
sl@0
    76
	iAvail=KNullPageRef;
sl@0
    77
	iDirty=EFalse;
sl@0
    78
	}
sl@0
    79
sl@0
    80
EXPORT_C void RStorePagePool::Open(CStreamStore& aStore,const TStorePagePoolToken& aToken)
sl@0
    81
/** Opens an existing pool.
sl@0
    82
sl@0
    83
@param aStore Stream store for the pool
sl@0
    84
@param aToken Pool settings */
sl@0
    85
	{
sl@0
    86
	iStore=&aStore;
sl@0
    87
	iHead=aToken.iHead;
sl@0
    88
	iAvail=aToken.iAvail;
sl@0
    89
	iDirty=EFalse;
sl@0
    90
	}
sl@0
    91
sl@0
    92
EXPORT_C TStorePagePoolToken RStorePagePool::Token() const
sl@0
    93
/** Gets an object that encapsulates the page pool settings.   
sl@0
    94
sl@0
    95
That object can then be used to externalise the settings.
sl@0
    96
sl@0
    97
@return Encapsulates the page pool settings */
sl@0
    98
	{
sl@0
    99
	return TStorePagePoolToken(iHead,iAvail);
sl@0
   100
	}
sl@0
   101
sl@0
   102
EXPORT_C void RStorePagePool::Close()
sl@0
   103
/** Flushes and purges the page cache and stops using the stream store. */
sl@0
   104
	{
sl@0
   105
	TCachePagePool::Flush();
sl@0
   106
	Release();
sl@0
   107
	}
sl@0
   108
sl@0
   109
EXPORT_C TBool RStorePagePool::ReclaimL()
sl@0
   110
/** Deletes the first stream that stores reclaimable pages in the store.
sl@0
   111
sl@0
   112
@return True if there are remaining streams (pages), false if there are no 
sl@0
   113
more streams */
sl@0
   114
	{
sl@0
   115
	__ASSERT_DEBUG(iStore!=NULL,Panic(EPageNotOpen));
sl@0
   116
	__ASSERT_ALWAYS(iAvail==KNullPageRef,Panic(EPageReclaimAvailable));
sl@0
   117
	if (iHead!=KNullStreamId)
sl@0
   118
		{
sl@0
   119
		RStoreReadStream out;
sl@0
   120
		out.OpenLC(*iStore,iHead);
sl@0
   121
		TStreamId next;
sl@0
   122
		out>>next;
sl@0
   123
		CleanupStack::PopAndDestroy();
sl@0
   124
		iStore->DeleteL(iHead);
sl@0
   125
		iHead=next;
sl@0
   126
		MarkDirty();
sl@0
   127
		if (next!=KNullStreamId)
sl@0
   128
			return ETrue;
sl@0
   129
		}
sl@0
   130
	return EFalse;
sl@0
   131
	}
sl@0
   132
sl@0
   133
EXPORT_C void RStorePagePool::ReclaimAllL()
sl@0
   134
/** Deletes all streams in the store. */
sl@0
   135
	{
sl@0
   136
	while (ReclaimL())
sl@0
   137
		;
sl@0
   138
	}
sl@0
   139
sl@0
   140
EXPORT_C TPageRef RStorePagePool::ExtendL(const TAny* aPage,TPageReclamation aReclamation)
sl@0
   141
//
sl@0
   142
// Create a new page.
sl@0
   143
//
sl@0
   144
	{
sl@0
   145
	return StorePagePool::ExtendL(*this,aPage,aReclamation);
sl@0
   146
	}
sl@0
   147
sl@0
   148
EXPORT_C void RStorePagePool::WriteL(TPageRef aRef,const TAny* aPage,TPageChange aChange)
sl@0
   149
//
sl@0
   150
// Write a page to the store.
sl@0
   151
//
sl@0
   152
	{
sl@0
   153
	StorePagePool::WriteL(*this,aRef,aPage,aChange);
sl@0
   154
	}
sl@0
   155
sl@0
   156
EXPORT_C void RStorePagePool::ReadL(TPageRef aRef,TAny* aPage)
sl@0
   157
//
sl@0
   158
// Read from the store into a page.
sl@0
   159
//
sl@0
   160
	{
sl@0
   161
	StorePagePool::ReadL(*this,aRef,aPage);
sl@0
   162
	}
sl@0
   163
sl@0
   164
EXPORT_C void RStorePagePool::DoDeleteL(TPageRef aRef)
sl@0
   165
//
sl@0
   166
// Delete from store.
sl@0
   167
//
sl@0
   168
	{
sl@0
   169
	StorePagePool::DeleteL(*this,aRef);
sl@0
   170
	}
sl@0
   171
sl@0
   172
// Class RSecureStorePagePool
sl@0
   173
sl@0
   174
sl@0
   175
sl@0
   176
EXPORT_C RSecureStorePagePool::RSecureStorePagePool(const CPBEncryptSet& aKey)
sl@0
   177
	: iKey(aKey)
sl@0
   178
	{}
sl@0
   179
sl@0
   180
EXPORT_C RSecureStorePagePool::RSecureStorePagePool(CPageCache& aCache, const CPBEncryptSet& aKey)
sl@0
   181
	: RStorePagePool(aCache),
sl@0
   182
	iKey(aKey)
sl@0
   183
	{}
sl@0
   184
sl@0
   185
/** Adds a new page to the pool.
sl@0
   186
sl@0
   187
@param aPage Data for the page
sl@0
   188
@param aReclamation Flags that define how allocated pages can be reclaimed
sl@0
   189
@return Reference to newly created page */
sl@0
   190
EXPORT_C TPageRef RSecureStorePagePool::ExtendL(const TAny* aPage,TPageReclamation aReclamation)
sl@0
   191
	{		
sl@0
   192
	return StorePagePool::ExtendL(*this,aPage,aReclamation);
sl@0
   193
	}
sl@0
   194
sl@0
   195
/** Writes data to a page in the pool.
sl@0
   196
sl@0
   197
@param aRef Reference to the page to which to write
sl@0
   198
@param aPage Data to write
sl@0
   199
@param aChange Flags that define how a page should be treated when it is unlocked 
sl@0
   200
for writing */
sl@0
   201
EXPORT_C void RSecureStorePagePool::WriteL(TPageRef aRef,const TAny* aPage,TPageChange aChange)
sl@0
   202
	{
sl@0
   203
	StorePagePool::WriteL(*this,aRef,aPage,aChange);
sl@0
   204
	}
sl@0
   205
sl@0
   206
/** Reads a specified page from the pool.
sl@0
   207
sl@0
   208
@param aRef Reference to page to read
sl@0
   209
@param aPage On return, the page data */
sl@0
   210
EXPORT_C void RSecureStorePagePool::ReadL(TPageRef aRef,TAny* aPage)
sl@0
   211
	{
sl@0
   212
	StorePagePool::ReadL(*this,aRef,aPage);
sl@0
   213
	}
sl@0
   214
sl@0
   215
/** Deletes a specified page.
sl@0
   216
sl@0
   217
@param aRef Reference to page to delete */
sl@0
   218
EXPORT_C void RSecureStorePagePool::DoDeleteL(TPageRef aRef)
sl@0
   219
	{
sl@0
   220
	StorePagePool::DeleteL(*this,aRef);
sl@0
   221
	}
sl@0
   222
sl@0
   223
// Class StorePagePool
sl@0
   224
// the implementation of the clever bits to share code between the secure and plain sotre page pools
sl@0
   225
sl@0
   226
const TInt KMaxPageIndex=15;
sl@0
   227
sl@0
   228
inline TInt PageSize(const CPBEncryptionBase* aKey)
sl@0
   229
	{
sl@0
   230
	return aKey ? aKey->MaxCiphertextLength(KPoolPageSize) : KPoolPageSize;
sl@0
   231
	}
sl@0
   232
sl@0
   233
inline TPageRef PageRef(TStreamId anId,TInt anIndex=0)
sl@0
   234
	{
sl@0
   235
	__ASSERT_DEBUG(anIndex>=0&&anIndex<=KMaxPageIndex,User::Invariant());
sl@0
   236
	return TPageRef((anId.Value()<<4)+anIndex);
sl@0
   237
	}
sl@0
   238
inline TInt PageIndex(TPageRef aRef)
sl@0
   239
	{return aRef.Value()&0xf;}
sl@0
   240
inline TStreamId StreamId(TPageRef aRef)
sl@0
   241
	{return TStreamId(aRef.Value()>>4);}
sl@0
   242
inline TStreamPos PagePos(TInt anIndex,TInt aPageSize)
sl@0
   243
	{return TStreamPos(sizeof(TStreamId)+(anIndex-1)*aPageSize);}
sl@0
   244
sl@0
   245
void StorePagePool::PadL(RWriteStream& aStream,TInt aLength)
sl@0
   246
//
sl@0
   247
// zero-pad a stream
sl@0
   248
//
sl@0
   249
	{
sl@0
   250
	TUint8 zero[KPoolPageSize];
sl@0
   251
	Mem::FillZ(zero,KPoolPageSize);
sl@0
   252
	while ((aLength-=KPoolPageSize)>0)
sl@0
   253
		aStream.WriteL(zero,KPoolPageSize);
sl@0
   254
	aStream.WriteL(zero,aLength+KPoolPageSize);
sl@0
   255
	}
sl@0
   256
sl@0
   257
//
sl@0
   258
// Encrypt the page to the stream, padding out to full page size
sl@0
   259
void StorePagePool::EncryptL(RWriteStream& aStream,const TAny* aPage,const CPBEncryptionBase& aKey)
sl@0
   260
	{
sl@0
   261
	REncryptStream encrypt;
sl@0
   262
	encrypt.OpenLC(aStream,aKey);
sl@0
   263
	encrypt.WriteL((const TUint8*)aPage,KPoolPageSize);
sl@0
   264
	encrypt.CommitL();
sl@0
   265
	CleanupStack::PopAndDestroy(&encrypt);
sl@0
   266
	}
sl@0
   267
sl@0
   268
//
sl@0
   269
// Encrypt the page to the stream, padding out to full page size
sl@0
   270
//
sl@0
   271
void StorePagePool::EncryptNewL(RWriteStream& aStream,const TAny* aPage,const CPBEncryptionBase& aKey)
sl@0
   272
	{
sl@0
   273
	TStreamPos pos=aStream.Sink()->TellL(MStreamBuf::EWrite);		// remember the start of the data
sl@0
   274
	EncryptL(aStream,aPage,aKey);
sl@0
   275
	TInt pad=PageSize(&aKey)-(aStream.Sink()->TellL(MStreamBuf::EWrite)-pos);		// bytes written
sl@0
   276
	__ASSERT_DEBUG(pad>=0,Panic(EPageCipherTextOverrun));
sl@0
   277
	if (pad)
sl@0
   278
		PadL(aStream,pad);
sl@0
   279
	}
sl@0
   280
sl@0
   281
//
sl@0
   282
// Encrypt the page to the stream, padding out to full page size
sl@0
   283
//
sl@0
   284
void StorePagePool::DecryptL(RReadStream& aStream,TAny* aPage,const CPBEncryptionBase& aKey)
sl@0
   285
	{
sl@0
   286
	RDecryptStream decrypt;
sl@0
   287
	decrypt.OpenLC(aStream,aKey);
sl@0
   288
	decrypt.ReadL((TUint8*)aPage,KPoolPageSize);
sl@0
   289
	CleanupStack::PopAndDestroy();
sl@0
   290
	}
sl@0
   291
sl@0
   292
void StorePagePool::SeekL(MStreamBuf* aBuf,TInt aMark,TPageRef aRef,const CPBEncryptionBase* aKey)
sl@0
   293
	{
sl@0
   294
	TInt ix=PageIndex(aRef);
sl@0
   295
	if (ix)
sl@0
   296
		aBuf->SeekL(aMark,PagePos(ix,PageSize(aKey)));
sl@0
   297
	}
sl@0
   298
sl@0
   299
TPageRef StorePagePool::ExtendL(RStorePagePool& aPool,const TAny* aPage,TPageReclamation aReclamation,const CPBEncryptionBase* aKey)
sl@0
   300
	{
sl@0
   301
	__ASSERT_DEBUG(aPool.iStore!=NULL,Panic(EPageNotOpen));
sl@0
   302
	TPageRef page;
sl@0
   303
	RStoreWriteStream out;
sl@0
   304
	if (aReclamation==EPageDeleteOnly)
sl@0
   305
		{	// non-reclaimable page
sl@0
   306
		page=PageRef(out.CreateLC(*aPool.iStore));
sl@0
   307
		}
sl@0
   308
	else if (aPool.iAvail!=KNullPageRef)
sl@0
   309
		{	// use free page
sl@0
   310
		page=aPool.iAvail;
sl@0
   311
		out.OpenLC(*aPool.iStore,StreamId(page));
sl@0
   312
		MStreamBuf* buf=out.Sink();
sl@0
   313
		SeekL(buf,MStreamBuf::ERead|MStreamBuf::EWrite,page,aKey);
sl@0
   314
		RReadStream in(buf);
sl@0
   315
		in>>aPool.iAvail;
sl@0
   316
		aPool.MarkDirty();
sl@0
   317
		}
sl@0
   318
	else
sl@0
   319
		{	// allocate new block, and create free list
sl@0
   320
		TStreamId id=out.CreateLC(*aPool.iStore);
sl@0
   321
		out<<aPool.iHead;			// block list
sl@0
   322
		const TInt pad=PageSize(aKey)-sizeof(TPageRef);
sl@0
   323
		page=KNullPageRef;	// free-list terminator
sl@0
   324
		for (TInt index=0;++index<KMaxPageIndex;)
sl@0
   325
			{
sl@0
   326
			out<<page;
sl@0
   327
			PadL(out,pad);
sl@0
   328
			page=PageRef(id,index);
sl@0
   329
			}
sl@0
   330
		aPool.MarkDirty();
sl@0
   331
		aPool.iHead=id;
sl@0
   332
		aPool.iAvail=page;
sl@0
   333
		page=PageRef(id,KMaxPageIndex);	// page is written at end
sl@0
   334
		}
sl@0
   335
	if (aKey)
sl@0
   336
		EncryptNewL(out,aPage,*aKey);
sl@0
   337
	else
sl@0
   338
		out.WriteL((const TUint8*)aPage,KPoolPageSize);
sl@0
   339
	out.CommitL();
sl@0
   340
	CleanupStack::PopAndDestroy();
sl@0
   341
	return page;
sl@0
   342
	}
sl@0
   343
sl@0
   344
void StorePagePool::WriteL(RStorePagePool& aPool,TPageRef aRef,const TAny* aPage,TPageChange aChange,const CPBEncryptionBase* aKey)
sl@0
   345
	{
sl@0
   346
	__ASSERT_DEBUG(aChange>=EPageDirty,Panic(EPageChangeInvalid));
sl@0
   347
	__ASSERT_DEBUG(aPool.iStore!=NULL,Panic(EPageNotOpen));
sl@0
   348
	RStoreWriteStream out;
sl@0
   349
	if (aChange==EPageUpdate)
sl@0
   350
		{
sl@0
   351
		__ASSERT_DEBUG(PageIndex(aRef)==0,Panic(EPageChangeInvalid));
sl@0
   352
		out.ReplaceLC(*aPool.iStore,StreamId(aRef));
sl@0
   353
		if (aKey)
sl@0
   354
			EncryptNewL(out,aPage,*aKey);
sl@0
   355
		else
sl@0
   356
			out.WriteL((const TUint8*)aPage,KPoolPageSize);
sl@0
   357
		}
sl@0
   358
	else
sl@0
   359
		{	// re-write the page, no padding req'd
sl@0
   360
		out.OpenLC(*aPool.iStore,StreamId(aRef));
sl@0
   361
		SeekL(out.Sink(),MStreamBuf::EWrite,aRef,aKey);
sl@0
   362
		if (aKey)
sl@0
   363
			EncryptL(out,aPage,*aKey);
sl@0
   364
		else
sl@0
   365
			out.WriteL((const TUint8*)aPage,KPoolPageSize);
sl@0
   366
		}
sl@0
   367
	out.CommitL();
sl@0
   368
	CleanupStack::PopAndDestroy();
sl@0
   369
	}
sl@0
   370
sl@0
   371
void StorePagePool::ReadL(RStorePagePool& aPool,TPageRef aRef,TAny* aPage,const CPBEncryptionBase* aKey)
sl@0
   372
	{
sl@0
   373
	__ASSERT_DEBUG(aPool.iStore!=NULL,Panic(EPageNotOpen));
sl@0
   374
	RStoreReadStream in;
sl@0
   375
	in.OpenLC(*aPool.iStore,StreamId(aRef));
sl@0
   376
	SeekL(in.Source(),MStreamBuf::ERead,aRef,aKey);
sl@0
   377
	if (aKey)
sl@0
   378
		DecryptL(in,aPage,*aKey);
sl@0
   379
	else
sl@0
   380
		in.ReadL((TUint8*)aPage,KPoolPageSize);
sl@0
   381
	CleanupStack::PopAndDestroy();	// in
sl@0
   382
	}
sl@0
   383
sl@0
   384
void StorePagePool::DeleteL(RStorePagePool& aPool,TPageRef aRef,const CPBEncryptionBase* aKey)
sl@0
   385
	{
sl@0
   386
	aPool.CacheDeleteL(aRef);
sl@0
   387
	__ASSERT_DEBUG(aPool.iStore!=NULL,Panic(EPageNotOpen));
sl@0
   388
	if (PageIndex(aRef)==0)
sl@0
   389
		{	//non-reclaimable
sl@0
   390
		aPool.iStore->DeleteL(StreamId(aRef));
sl@0
   391
		return;
sl@0
   392
		}
sl@0
   393
	// add to free list
sl@0
   394
	RStoreWriteStream out;
sl@0
   395
	out.OpenLC(*aPool.iStore,StreamId(aRef));
sl@0
   396
	SeekL(out.Sink(),MStreamBuf::EWrite,aRef,aKey);
sl@0
   397
	out<<aPool.iAvail;
sl@0
   398
	out.CommitL();
sl@0
   399
	CleanupStack::PopAndDestroy();
sl@0
   400
	aPool.iAvail=aRef;
sl@0
   401
	aPool.MarkDirty();
sl@0
   402
	}