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.
18 // Class CDbStoreRecords::TIteratorC
20 class CDbStoreRecords::TIteratorC
22 friend class CDbStoreRecords;
24 inline TDbRecordId Current() const;
30 inline TDbRecordId CDbStoreRecords::TIteratorC::Current() const
34 // Class CDbStoreRecords::CIter
36 NONSHARABLE_CLASS(CDbStoreRecords::CIter) : public CDbRecordIter
39 CIter(CDbStoreRecords& aRecords);
41 inline CDbStoreRecords& Records() const;
44 TDbRecordId CurrentL();
45 TBool GotoL(TDbPosition aPosition);
46 TBool GotoL(TDbRecordId aRecordId,RDbTableRow& aBuffer);
47 TBool SeekL(const TDbLookupKey& aKey,RDbTable::TComparison aComparison);
48 TDeleted DoDeletedL(TDbPosition aPosition,TDbRecordId aRecordId,const RDbTableRow* aRow);
50 CDbStoreRecords::TIteratorC iIter;
53 CDbStoreRecords::CIter::CIter(CDbStoreRecords& aRecords)
54 : CDbRecordIter(aRecords)
57 inline CDbStoreRecords& CDbStoreRecords::CIter::Records() const
58 {return STATIC_CAST(CDbStoreRecords&,Host());}
60 TInt CDbStoreRecords::CIter::Count() const
62 return Records().Count();
65 TDbRecordId CDbStoreRecords::CIter::CurrentL()
67 return iIter.Current();
70 TBool CDbStoreRecords::CIter::GotoL(TDbPosition aPosition)
72 return Records().GotoL(aPosition,iIter);
75 TBool CDbStoreRecords::CIter::GotoL(TDbRecordId aRecordId,RDbTableRow&)
77 return Records().GotoL(aRecordId,iIter);
80 TBool CDbStoreRecords::CIter::SeekL(const TDbLookupKey&,RDbTable::TComparison)
82 // Cannot do this on a table iterator
89 CDbStoreRecords::CIter::TDeleted CDbStoreRecords::CIter::DoDeletedL(TDbPosition aPosition,TDbRecordId,const RDbTableRow*)
91 // reposition to next after a record is deleted
92 // Previous only required for reversed (index) iterators
95 return Records().DeletedL(aPosition,iIter) ? EAtRow : ENoRow;
99 // Class CDbStoreRecords::TToken
101 void CDbStoreRecords::TToken::ExternalizeL(RWriteStream& aStream) const
103 aStream<<iHead<<iNext.Value()<<TCardinality(iCount)<<TUint32(iAutoIncrement);
106 void CDbStoreRecords::TToken::InternalizeL(RReadStream& aStream)
109 iNext=aStream.ReadUint32L();
113 iAutoIncrement=aStream.ReadUint32L();
117 // Class CDbStoreRecords
119 CDbStoreRecords::CDbStoreRecords(CClusterCache& aCache)
123 CDbStoreRecords::~CDbStoreRecords()
128 TStreamId CDbStoreRecords::CreateL(CClusterCache& aCache)
130 // Create a new record space in the store, do not create a records object
134 token.iHead=ClusterId(aCache.Store().ExtendL());
135 aCache.ClusterL().Create(token.iHead);
136 token.iNext=RecordId(token.iHead,0);
138 token.iAutoIncrement=0;
139 RStoreWriteStream strm;
140 TStreamId id=strm.CreateLC(aCache.Store());
143 CleanupStack::PopAndDestroy();
147 CDbStoreRecords* CDbStoreRecords::NewL(CClusterCache& aCache,const CDbStoreDef& aDef)
149 // Create a record space
152 CDbStoreRecords* self=new(ELeave) CDbStoreRecords(aCache);
153 CleanupStack::PushL(self);
154 self->iClustering=aDef.Clustering();
155 self->iTokenId=aDef.TokenId();
160 TBool CDbStoreRecords::RestoreL()
162 // Restore an existing record space from the store
165 RStoreReadStream strm;
166 strm.OpenLC(iCache.Store(),iTokenId);
168 CleanupStack::PopAndDestroy();
170 iMap.ResetL(iToken.iHead);
174 void CDbStoreRecords::DestroyL()
176 // Destroy the record space
179 iCache.Store().DeleteL(iTokenId);
182 TInt CDbStoreRecords::CardinalityL(CStreamStore& aStore,const CDbStoreDef& aDef)
184 // Return the record count without constructing the entire table
187 RStoreReadStream strm;
188 strm.OpenLC(aStore,aDef.TokenId());
191 CleanupStack::PopAndDestroy();
195 void CDbStoreRecords::SynchL()
197 // write persistent token to the store
200 RStoreWriteStream strm;
201 strm.ReplaceLC(iCache.Store(),iTokenId);
204 CleanupStack::PopAndDestroy();
207 TInt CDbStoreRecords::DiscardL(TClusterId& aCluster)
209 // discard the cluster as part of the incremental drop
210 // aCluster is updated to the next cluster, the number of records contained is returned
215 CCluster* cluster=iCache.Cluster(aCluster);
218 iCache.Store().DeleteL(aCluster);
221 for (TUint members=des.iMembership;members;members&=members-1)
226 TClusterId CDbStoreRecords::AlterL(TClusterId aCluster,CCluster::MAlter& aAlterer)
228 CCluster& cluster=iCache.ClusterL(aCluster);
229 cluster.AlterL(aAlterer);
230 return cluster.Des().iNext;
233 TPtrC8 CDbStoreRecords::ReadL(TDbRecordId aRecordId) const
235 return iCache.ClusterL(ClusterId(aRecordId)).RecordL(RecordIndex(aRecordId));
238 TUint CDbStoreRecords::AutoIncrementL()
240 // Provide the next value for an auto-increment column
243 return iToken.iAutoIncrement++;
246 TUint8* CDbStoreRecords::UpdateRecordL(TDbRecordId aRecordId,TInt aNewSize)
248 // read the cluster and return a writable descriptor over the new record data
251 return iCache.ClusterL(ClusterId(aRecordId)).UpdateL(RecordIndex(aRecordId),aNewSize);
255 TUint8* CDbStoreRecords::DoNewL(TInt aRecordSize)
257 // Phase 1 of appending a records
260 return UpdateRecordL(iToken.iNext,aRecordSize);
263 TDbRecordId CDbStoreRecords::AppendL()
265 // Phase 2 of appending a record
268 TDbRecordId id=iToken.iNext;
269 TClusterId clusterId=ClusterId(id);
270 CCluster* cluster=iCache.Cluster(clusterId);
272 TInt nextIndex=RecordIndex(id)+1;
273 if (nextIndex>=iClustering || cluster->IsFull())
275 TClusterId newcluster=ClusterId(iCache.Store().ExtendL());
276 cluster->Relink(newcluster);
278 cluster->Create(newcluster);
279 iMap.BindL(clusterId,newcluster);
280 iLinks.Bind(clusterId,newcluster,iMap);
281 iToken.iNext=RecordId(newcluster,0);
284 iToken.iNext=RecordId(clusterId,nextIndex);
289 TUint8* CDbStoreRecords::DoReplaceL(TDbRecordId aRecordId,TInt aRecordSize)
291 return UpdateRecordL(aRecordId,aRecordSize);
294 void CDbStoreRecords::DoEraseL(TDbRecordId aRecordId)
296 TClusterId clusterId=ClusterId(aRecordId);
297 CCluster& cluster=iCache.ClusterL(clusterId);
298 if (!cluster.DeleteL(RecordIndex(aRecordId)) && clusterId!=ClusterId(iToken.iNext))
299 { // cluster is now empty, but don't drop the last cluster, coz it hasn't all been used!
301 TClusterId prev=PreviousClusterL(des,clusterId);
302 TClusterId next=cluster.Des().iNext; // next cluster
303 cluster.Discard(); // discard the cluster
304 iCache.Store().DeleteL(clusterId);
305 if (prev!=KNullClusterId)
306 iCache.ClusterL(prev).Relink(next);
309 iLinks.Drop(clusterId,next);
310 iMap.DropL(clusterId,next);
315 CDbRecordIter* CDbStoreRecords::IteratorL()
317 return new(ELeave) CIter(*this);
320 void CDbStoreRecords::CompleteMapL()
322 TClusterId cluster=iMap.LastBound();
325 do cluster=NextClusterL(des,cluster); while (cluster!=KNullClusterId);
328 void CDbStoreRecords::DesL(TClusterDes& aDes,TClusterId aCluster)
330 // Read just the cluster descriptor
333 CCluster* cluster=iCache.Cluster(aCluster);
338 RStoreReadStream stream;
339 stream.OpenLC(iCache.Store(),aCluster);
341 CleanupStack::PopAndDestroy();
345 TClusterId CDbStoreRecords::NextClusterL(TClusterDes& aDes,TClusterId aCluster)
347 TClusterId next=aDes.iNext;
348 if (next==KNullClusterId)
349 iMap.Complete(aCluster);
352 iMap.BindL(aCluster,next);
353 iLinks.Bind(aCluster,next,iMap);
359 TBool CDbStoreRecords::LocateL(TClusterId aCluster)
361 // Locate the cluster in the table. If not present return EFalse
362 // If present fill up the cluster link cache with the loop
363 // containing the predecessor to aCluster
364 // aDes will have the previous cluster des
367 TClusterId cluster=aCluster;
368 __ASSERT(aCluster!=iToken.iHead);
369 __ASSERT(!iLinks.At(aCluster,cluster));
371 if (!iMap.IsComplete())
375 TClusterId links[RClusterMap::ESeparation];
377 for (TInt n=RClusterMap::ESeparation;n>0;--n)
380 TBool r=iMap.At(cluster,cluster);
384 __ASSERT(cluster!=KNullClusterId); // only iHead->Null
385 iLinks.Reset(cluster);
386 while (aCluster!=des.iNext)
387 cluster=NextClusterL(des,cluster);
393 return EFalse; // not in this table!
396 TClusterId CDbStoreRecords::PreviousClusterL(TClusterDes& aDes,TClusterId aCluster)
398 if (aCluster==iToken.iHead)
399 return KNullClusterId;
400 if (!iLinks.At(aCluster,aCluster))
402 __DEBUG(TBool dbgchk=) LocateL(aCluster);
404 __DEBUG(dbgchk=) iLinks.At(aCluster,aCluster);
411 TBool CDbStoreRecords::GotoL(TDbPosition aPosition,TIteratorC& anIterator)
413 TClusterId cluster=ClusterId(anIterator.iCurrent);
414 TInt index=RecordIndex(anIterator.iCurrent);
420 DesL(anIterator.iDes,cluster=iToken.iHead);
421 iLinks.Reset(cluster);
423 // drop through to next
427 TUint membership=anIterator.iDes.iMembership;
428 while (++index<KMaxClustering)
430 if ((membership>>index)&1)
432 __ASSERT(cluster!=ClusterId(iToken.iNext)||index<RecordIndex(iToken.iNext));
433 anIterator.iCurrent=RecordId(cluster,index);
437 cluster=NextClusterL(anIterator.iDes,cluster);
438 if (cluster==KNullClusterId)
439 return EFalse; // ran out of data
443 DesL(anIterator.iDes,cluster=ClusterId(iToken.iNext));
444 index=KMaxClustering;
445 // drop through to previous
449 TUint membership=anIterator.iDes.iMembership;
452 if ((membership>>index)&1)
454 anIterator.iCurrent=RecordId(cluster,index);
459 cluster=PreviousClusterL(anIterator.iDes,cluster);
460 if (cluster==KNullClusterId)
461 return EFalse; // ran out of data
462 index=KMaxClustering;
467 TBool CDbStoreRecords::DeletedL(TDbPosition aPosition,TIteratorC& anIterator)
469 // Record has been deleted
472 anIterator.iDes.iMembership&=~(1<<RecordIndex(anIterator.iCurrent));
473 return GotoL(aPosition,anIterator);
476 TBool CDbStoreRecords::GotoL(TDbRecordId aRecordId,TIteratorC& anIterator)
478 // Set the iterator to the record id, return false if the record is not present
481 TClusterId cluster=ClusterId(aRecordId);
482 if (cluster!=iToken.iHead && !iLinks.Has(cluster) && !LocateL(cluster))
484 anIterator.iCurrent=aRecordId;
485 DesL(anIterator.iDes,cluster);
486 return (anIterator.iDes.iMembership>>RecordIndex(aRecordId))&1;
489 TBool CDbStoreRecords::ExistsL(TDbRecordId aRecordId)
491 // Ensure that the record is in this table
495 return GotoL(aRecordId,iter);
498 // Class HUnicodeCompressor
500 NONSHARABLE_CLASS(HUnicodeCompressor) : public TStreamFilter
503 HUnicodeCompressor(MStreamBuf* aSink);
507 TInt Capacity(TInt aMaxLength);
508 TInt FilterL(TAny* aPtr,TInt aMaxLength,const TUint8*& aFrom,const TUint8* anEnd);
510 enum {EFlushBufferSize=16};
512 TUnicodeCompressor iCompressor;
515 HUnicodeCompressor::HUnicodeCompressor(MStreamBuf* aSink)
517 Set(aSink,EAttached|EWrite);
520 void HUnicodeCompressor::DoRelease()
522 TStreamFilter::DoRelease();
526 TInt HUnicodeCompressor::Capacity(TInt aMaxLength)
528 // Return the maximum guaranteed input used for aMaxLength output.
529 // SUC at worst expands n chars to 3n bytes
532 aMaxLength=(aMaxLength+2)/3; // # chars input guaranteed
533 return aMaxLength*2; // # bytes
536 TInt HUnicodeCompressor::FilterL(TAny* aPtr,TInt aMaxLength,const TUint8*& aFrom,const TUint8* aEnd)
538 TMemoryUnicodeSource source(reinterpret_cast<const TUint16*>(aFrom));
540 iCompressor.CompressL(reinterpret_cast<TUint8*>(aPtr),source,aMaxLength,(aEnd-aFrom)>>1,&aMaxLength,&used);
545 void HUnicodeCompressor::DoSynchL()
550 TUint8 buf[EFlushBufferSize];
552 iCompressor.FlushL(buf,EFlushBufferSize,emit);
556 TStreamFilter::DoSynchL();
560 // Class HUnicodeExander
562 NONSHARABLE_CLASS(HUnicodeExpander) : public TStreamFilter
565 HUnicodeExpander(MStreamBuf* aSource);
569 TInt Capacity(TInt aMaxLength);
570 TInt FilterL(TAny* aPtr,TInt aMaxLength,const TUint8*& aFrom,const TUint8* anEnd);
572 enum {EFlushBufferSize=16};
574 TUnicodeExpander iExpander;
577 HUnicodeExpander::HUnicodeExpander(MStreamBuf* aSource)
579 Set(aSource,EAttached|ERead);
582 void HUnicodeExpander::DoRelease()
584 TStreamFilter::DoRelease();
588 TInt HUnicodeExpander::Capacity(TInt aMaxLength)
590 // Return the maximum guaranteed input used for aMaxLength output.
591 // SUC at worst expands n chars to 3n bytes
594 return aMaxLength>>1; // best expansion from ASCII chars
597 TInt HUnicodeExpander::FilterL(TAny* aPtr,TInt aMaxLength,const TUint8*& aFrom,const TUint8* aEnd)
599 TMemoryUnicodeSink sink(reinterpret_cast<TUint16*>(aPtr));
601 iExpander.ExpandL(sink,aFrom,aMaxLength>>1,aEnd-aFrom,&aMaxLength,&used);
603 return aMaxLength<<1;
607 void HUnicodeExpander::DoSynchL()
612 // TUint8 buf[EFlushBufferSize];
614 // iCompressor.FlushL(buf,EFlushBufferSize,&emit);
618 TStreamFilter::DoSynchL();
623 // Class CDbStoreBlobs
625 CDbStoreBlobs::CDbStoreBlobs(CDbStoreDatabase& aDatabase,TInt aInlineLimit)
626 : iDatabase(aDatabase)
628 SetInlineLimit(aInlineLimit);
631 MStreamBuf* CDbStoreBlobs::DoCreateL(TDbBlobId& aBlobId,TDbColType aType)
633 __ASSERT(TDbCol::IsLong(aType));
635 RDbStoreWriteStream strm(iDatabase);
636 aBlobId=strm.CreateLC(iDatabase.Store()).Value();
637 strm.FilterL(aType!=EDbColLongBinary?strm.EText:strm.EBinary,aBlobId);
638 MStreamBuf* blob=strm.Sink();
639 if (aType==EDbColLongText16)
640 blob=new(ELeave) HUnicodeCompressor(blob);
645 MStreamBuf* CDbStoreBlobs::ReadL(TDbBlobId aBlobId,TDbColType aType) const
647 __ASSERT(TDbCol::IsLong(aType));
649 RDbStoreReadStream strm(iDatabase);
650 strm.OpenLC(iDatabase.Store(),aBlobId);
651 strm.FilterL(aType!=EDbColLongBinary?strm.EText:strm.EBinary,aBlobId);
652 MStreamBuf* blob=strm.Source();
653 if (aType==EDbColLongText16)
654 blob=new(ELeave) HUnicodeExpander(blob);
659 void CDbStoreBlobs::DoDeleteL(TDbBlobId aBlobId)
661 iDatabase.Store().DeleteL(TStreamId(aBlobId));