First public contribution.
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.
19 // Class TDbStoreIndexStats::TBound
21 inline void TDbStoreIndexStats::TBound::Set(TReal64 aBound)
24 void TDbStoreIndexStats::TBound::Set(const TInt64& aBound)
25 {iValue=I64REAL(aBound);}
27 void TDbStoreIndexStats::TBound::Set(const TText8* aPtr,TInt aLen,const TTextOps& aConv)
30 const TText8* const end=aPtr+aLen;
36 TInt64 t(aConv.Fold(*aPtr++));
39 } while ((shift-=8)>=0);
43 void TDbStoreIndexStats::TBound::Set(const TText16* aPtr,TInt aLen,const TTextOps& aConv)
46 const TText16* const end=aPtr+aLen;
52 TInt64 t(aConv.Fold(*aPtr++));
55 } while ((shift-=16)>=0);
59 void TDbStoreIndexStats::TBound::Set(const TDbLookupKey::SColumn& aBound,const TTextOps& aConv)
69 Set(aBound.iTime().Int64());
75 Set(aBound.iDes8.iPtr,aBound.iDes8.iLength,aConv);
78 Set(aBound.iDes16.iPtr,aBound.iDes16.iLength,aConv);
83 // Class TDbStoreIndexStats
85 inline TBool TDbStoreIndexStats::NeedsRefresh() const
87 return iRefresh<=ERefresh;
90 inline TInt TDbStoreIndexStats::Cardinality() const
95 // update the refresh count
96 inline void TDbStoreIndexStats::Touch()
103 // an entry is inserted
104 inline void TDbStoreIndexStats::Inc()
110 // an entry is removed
111 inline void TDbStoreIndexStats::Dec()
117 // contents have changed radically. provoke an immediate refresh
118 inline void TDbStoreIndexStats::Reset(TInt aCardinality)
120 iCardinality=aCardinality;
124 // stats have been refreshed. set for next update
125 inline void TDbStoreIndexStats::Refresh(TDbStoreIndexStats::TType aType)
127 iFlags=(iFlags&~EFlgDiscrete)|aType;
128 iRefresh=(iCardinality>>ERefreshFactor)+1;
132 // Internalize the index statistics
133 // This must handle a stream externalized by builds before 052
135 void TDbStoreIndexStats::InternalizeL(RReadStream& aStream)
140 TUint refresh=aStream.ReadUint32L();
141 iFlags=refresh&EFlagsMask;
142 iRefresh=TInt(refresh>>ERefreshShift)+EInvalid;
144 // pre-build-052 data would run out here
145 // if there is no more data, mark the (non-cardinality) stats as invalid
147 TRAPD(r,aStream>>iLower.iValue>>iUpper.iValue;)
149 iRefresh=EInvalid; // mark as invalid data
151 __LEAVE_IF_ERROR(r); // just an "ordinary" error
155 // Notes: iCardinality cannot exceed 2^29 (due to storage mechanism)
156 // thus iRefresh cannot exceed 2^(29-ERefreshFactor)+1
157 // this leaves the 5 m.s.b. clear in iRefresh-EInvalid
158 // the flags bits are stored there
160 void TDbStoreIndexStats::ExternalizeL(RWriteStream& aStream) const
162 __ASSERT((iFlags&~EFlagsMask)==0);
163 aStream<<TCardinality(iCardinality);
164 aStream<<TUint32((TUint(iRefresh-EInvalid)<<ERefreshShift)|iFlags);
165 aStream<<iLower.iValue<<iUpper.iValue;
169 // Reverse the span and bounds
171 TInt TDbStoreIndexStats::ReverseSpan(TUint aInclusion,const TDbLookupKey* aLower,const TDbLookupKey* aUpper,const TTextOps& aConv) const
173 __ASSERT(iLower.iValue>iUpper.iValue);
174 TDbStoreIndexStats stats(*this);
177 return stats.Span((aInclusion<<1)|(aInclusion>>1),aUpper,aLower,aConv);
181 // Evaluate the probable proportion of the index set contained within the bounds
183 TInt TDbStoreIndexStats::Span(TUint aInclusion,const TDbLookupKey* aLower,const TDbLookupKey* aUpper,const TTextOps& aConv) const
186 return CDbTable::EUnavailableSpan; // No valid index data
188 return 0; // no rows at all
190 if (iLower.iValue>iUpper.iValue)
191 return ReverseSpan(aInclusion,aLower,aUpper,aConv); // descending index
193 aInclusion&=CDbRecordIndex::EIncludeBoth;
198 bound.Set(*aLower->First(),aConv);
199 if (bound.iValue>=l.iValue)
202 aInclusion|=CDbRecordIndex::EIncludeLower;
205 aInclusion|=CDbRecordIndex::EIncludeLower;
211 bound.Set(*aUpper->First(),aConv);
212 if (bound.iValue<=h.iValue)
215 aInclusion|=CDbRecordIndex::EIncludeUpper;
218 aInclusion|=CDbRecordIndex::EIncludeUpper;
220 TRealX restrict(h.iValue);
221 restrict.SubEq(l.iValue); // extended precision--no errors
222 TRealX span(iUpper.iValue);
223 span.SubEq(iLower.iValue);
224 if (iFlags&EFlgDiscrete)
228 else if (aInclusion==CDbRecordIndex::EIncludeBoth)
232 else if (restrict.IsZero()) // single value continuous range
234 return (span.IsZero() && aInclusion==CDbRecordIndex::EIncludeBoth)
235 ? CDbTable::EFullIndexSpan : 0;
237 if (restrict<=TRealX(0))
238 return 0; // no overlap
239 restrict.DivEq(span);
240 restrict.MultEq(TInt(CDbTable::EFullIndexSpan));
241 return TInt(restrict);
245 // Class CDbStoreIndex
247 inline CDbStoreIndex::HKey& CDbStoreIndex::Key() const
252 inline const TBtree& CDbStoreIndex::Tree() const
257 inline TInt CDbStoreIndex::Count() const
259 return iStats.Cardinality();
263 // Class CDbStoreIndex::HKey
265 NONSHARABLE_CLASS(CDbStoreIndex::HKey) : public MBtreeKey
268 static HKey* NewL(const CDbKey& aKey,const HDbColumnSet& aColSet);
270 inline TInt EntrySize() const;
271 virtual TInt KeySize() const;
272 virtual TBool IncompleteKey() const;
274 TInt EntryL(TAny* aPtr,const RDbTableRow& aRow,TDbRecordId aRecord);
275 TInt EntryL(TAny* aPtr,const TDbLookupKey& aKey);
276 inline const TAny* Restriction() const;
277 inline void Restrict(const TAny* aRestriction);
279 void Bound(TDbStoreIndexStats::TBound& aBound,const TAny* aEntry);
280 TDbStoreIndexStats::TType KeyType() const;
282 const TAny* Key(const TAny* aRecord) const;
283 TInt Compare(const TAny* aLeft,const TAny* aRight) const;
286 inline TBool FullComparison() const;
288 void Between(const TAny* aLeft,const TAny* aRight,TBtreePivot& aPivot) const;
297 const TTextOps* iTextOps;
299 const SKeyCol* iEndOfKeys;
300 const SKeyCol* iRestrictedEndOfKeys;
301 SKeyCol iKeys[1]; // one or more keys
304 NONSHARABLE_CLASS(CDbStoreIndex::HDupKey) : public CDbStoreIndex::HKey
306 friend class CDbStoreIndex::HKey;
309 TInt KeySize() const;
310 const TAny* Key(const TAny* aRecord) const;
311 TInt Compare(const TAny* aLeft,const TAny* aRight) const;
312 TBool IncompleteKey() const;
315 inline TInt CDbStoreIndex::HKey::EntrySize() const
320 inline const TAny* CDbStoreIndex::HKey::Restriction() const
322 return iRestrictedEndOfKeys;
325 inline void CDbStoreIndex::HKey::Restrict(const TAny* aRestriction)
327 __ASSERT(aRestriction==0||(aRestriction>=iKeys&&aRestriction<=iEndOfKeys));
328 iRestrictedEndOfKeys=(const SKeyCol*)aRestriction;
331 inline TBool CDbStoreIndex::HKey::FullComparison() const
333 return iRestrictedEndOfKeys==NULL;
337 // Construct the key based on the key definition (must be valid for the table)
338 // and the column set provided
340 CDbStoreIndex::HKey* CDbStoreIndex::HKey::NewL(const CDbKey& aKey,const HDbColumnSet& aColumns)
342 TInt count=aKey.Count();
343 HKey* self=(HKey*)User::AllocLC(_FOFF(HKey,iKeys[count]));
348 self->iTextOps=&TTextOps::Ops(aKey.Comparison());
349 self->iEndOfKeys=&self->iKeys[count];
350 self->iRestrictedEndOfKeys=NULL;
351 TInt len=sizeof(TDbRecordId);
352 SKeyCol* pKey=&self->iKeys[0];
353 for (TInt ii=0;ii<count;++pKey,++ii)
355 const TDbKeyCol& key=aKey[ii];
356 pKey->iOrder=TUint8(key.iOrder);
357 pKey->iOrdinal=aColumns.ColNoL(key.iName);
358 if (pKey->iOrdinal==KDbNullColNo)
359 __LEAVE(KErrCorrupt);
360 const TDbColumnDef& col=aColumns[pKey->iOrdinal];
361 pKey->iType=col.iType;
362 pKey->iSize=CDbStoreIndexDef::KeySize(key,col);
363 len+=Align4(pKey->iSize);
366 if (self->KeySize()>KMaxIndexKeySize)
367 __LEAVE(KErrCorrupt);
373 // Construct an entry at aPtr from the record given
375 TInt CDbStoreIndex::HKey::EntryL(TAny* aPtr,const RDbTableRow& aRow,TDbRecordId aRecord)
377 TUint8* ptr=(TUint8*)aPtr;
378 Mem::FillZ(ptr,iSize);
379 *REINTERPRET_CAST(TDbRecordId*,ptr)=aRecord;
380 ptr+=sizeof(TDbRecordId);
381 const SKeyCol* const end=iEndOfKeys;
382 for (const SKeyCol* key=&iKeys[0];key<end;++key)
384 const TDbCell* cell=aRow.ColCell(key->iOrdinal);
385 TInt size=key->iSize;
386 if (cell->Length()!=0)
388 #ifdef __DOUBLE_WORDS_SWAPPED__
389 if (key->iType==EDbColReal64)
391 const TUint32* data=(TUint32*)cell->Data();
392 ((TUint32*)ptr)[0]=data[1];
393 ((TUint32*)ptr)[1]=data[0];
397 if (TDbCol::IsLong(TDbColType(key->iType)))
399 const TDbBlob& blob=*(const TDbBlob*)cell->Data();
401 Mem::Copy(ptr,blob.Data(),Min(size,blob.Size()));
404 aRow.Table().BlobsL()->ReadLC(blob.Id(),TDbColType(key->iType))->ReadL(ptr,size);
405 CleanupStack::PopAndDestroy(); // stream buffer
409 Mem::Copy(ptr,cell->Data(),Min(size,cell->Length()));
413 iRestrictedEndOfKeys=NULL; // use the full key for comparison
418 // Construct an entry from a lookup key
420 TInt CDbStoreIndex::HKey::EntryL(TAny* aPtr,const TDbLookupKey& aKey)
422 TUint8* ptr=(TUint8*)aPtr;
423 Mem::FillZ(ptr,iSize);
424 ptr+=sizeof(TDbRecordId);
425 const TDbLookupKey::SColumn* lkey=aKey.First();
426 const TDbLookupKey::SColumn* const lend=lkey+aKey.Count();
428 const SKeyCol* const end=iEndOfKeys;
429 const SKeyCol* key=&iKeys[0];
432 TDbColType ltype=lkey->iType;
433 TInt size=key->iSize;
442 if (ltype==EDbColUint32)
444 *(TUint32*)ptr=lkey->iUint32;
447 else if (ltype==EDbColInt32)
449 if (lkey->iInt32>=0) // domain check, unsigned value
451 *(TUint32*)ptr=lkey->iInt32;
455 else if (ltype==EDbColInt64)
457 const TInt64& v=lkey->iInt64;
458 if (I64HIGH(v)==0) // domain check, in unsigned 32-bit range
460 *(TUint32*)ptr=I64LOW(v);
469 if (ltype==EDbColInt32)
471 *(TInt32*)ptr=lkey->iInt32;
474 else if (ltype==EDbColUint32)
476 if (lkey->iUint32<=TUint(KMaxTInt)) // domain check, in signed range
478 *(TInt32*)ptr=lkey->iUint32;
482 else if (ltype==EDbColInt64)
484 const TInt64& v=lkey->iInt64;
487 if (h==(l>>31)) // domain check, in signed 32-bit range
496 if (ltype==EDbColReal32)
498 *(TReal32*)ptr=lkey->iReal32;
501 else if (ltype==EDbColReal64)
502 { // convert to TReal32, storing +-#inf if reqd.
503 TRealX(lkey->iReal64).GetTReal(*(TReal32*)ptr);
509 #ifdef __DOUBLE_WORDS_SWAPPED__
510 if (ltype==EDbColReal64)
512 const TUint32* data=(TUint32*)&lkey->iReal64;
513 ((TUint32*)ptr)[0]=data[1];
514 ((TUint32*)ptr)[1]=data[0];
521 if (ltype==key->iType)
522 Mem::Copy(ptr,&lkey->iInt64,size); // all at same address
527 case EDbColLongText8:
528 if (ltype==EDbColText8)
529 Mem::Copy(ptr,lkey->iDes8.iPtr,Min(size,lkey->iDes8.iLength));
534 case EDbColLongText16:
535 if (ltype==EDbColText16)
536 Mem::Copy(ptr,lkey->iDes16.iPtr,Min(size,lkey->iDes16.iLength<<1));
543 break; // end of lookup key
545 __LEAVE(KErrArgument); // too many keys
548 iRestrictedEndOfKeys=key; // use only the keys in the lookup for comparison
552 void CDbStoreIndex::HKey::Bound(TDbStoreIndexStats::TBound& aBound,const TAny* aEntry)
554 aEntry=PtrAdd(aEntry,sizeof(TDbRecordId)); // get to real key data
555 switch (iKeys[0].iType)
563 aBound.Set(TInt64(TUint(*STATIC_CAST(const TUint32*,aEntry))));
568 aBound.Set(TInt64(TInt(*STATIC_CAST(const TInt32*,aEntry))));
571 aBound.Set(*STATIC_CAST(const TInt64*,aEntry));
574 aBound.Set(STATIC_CAST(const TTime*,aEntry)->Int64());
577 aBound.Set(TReal64(*STATIC_CAST(const TReal32*,aEntry)));
580 #if !defined(__DOUBLE_WORDS_SWAPPED__)
581 aBound.Set(*STATIC_CAST(const TReal64*,aEntry));
585 ((TUint32*)&xKey)[0]=STATIC_CAST(const TUint32*,aEntry)[1];
586 ((TUint32*)&xKey)[1]=STATIC_CAST(const TUint32*,aEntry)[0];
592 case EDbColLongText8:
593 aBound.Set(STATIC_CAST(const TUint8*,aEntry),iKeys[0].iSize,*iTextOps);
596 case EDbColLongText16:
597 aBound.Set(STATIC_CAST(const TUint16*,aEntry),iKeys[0].iSize>>1,*iTextOps);
602 // Is the index key discrete or continous?
603 inline TDbStoreIndexStats::TType CDbStoreIndex::HKey::KeyType() const
605 return iKeys[0].iType==EDbColReal32 || iKeys[0].iType==EDbColReal64
606 ? TDbStoreIndexStats::EContinuous : TDbStoreIndexStats::EDiscrete;
609 TInt CDbStoreIndex::HKey::KeySize() const
611 return EntrySize()-sizeof(TDbRecordId);
615 // Report 'true' if the lookup key is not the entire B+tree key
616 // For a unique index this is if there is a restriction to less than the full key
618 TBool CDbStoreIndex::HKey::IncompleteKey() const
620 return iRestrictedEndOfKeys!=0 && iRestrictedEndOfKeys!=iEndOfKeys;
624 // For unique keys, key is after record id
626 const TAny* CDbStoreIndex::HKey::Key(const TAny* aRecord) const
628 return PtrAdd(aRecord,sizeof(TDbRecordId));
632 // compare the key part of the entry
634 TInt CDbStoreIndex::HKey::Compare(const TAny* aLeft,const TAny* aRight) const
636 const SKeyCol* end=iRestrictedEndOfKeys;
639 const SKeyCol* key=&iKeys[0];
642 TInt size=key->iSize;
652 rr=Comp::Compare(*STATIC_CAST(const TUint32*,aLeft),*STATIC_CAST(const TUint32*,aRight));
657 rr=Comp::Compare(*STATIC_CAST(const TInt32*,aLeft),*STATIC_CAST(const TInt32*,aRight));
660 rr=Comp::Compare(*STATIC_CAST(const TInt64*,aLeft),*STATIC_CAST(const TInt64*,aRight));
663 rr=Comp::Compare(*STATIC_CAST(const TTime*,aLeft),*STATIC_CAST(const TTime*,aRight));
666 rr=Comp::Compare(*STATIC_CAST(const TReal32*,aLeft),*STATIC_CAST(const TReal32*,aRight));
669 #if !defined(__DOUBLE_WORDS_SWAPPED__)
670 rr=Comp::Compare(*STATIC_CAST(const TReal64*,aLeft),*STATIC_CAST(const TReal64*,aRight));
673 ((TUint32*)&xLeft)[0]=STATIC_CAST(const TUint32*,aLeft)[1];
674 ((TUint32*)&xLeft)[1]=STATIC_CAST(const TUint32*,aLeft)[0];
676 ((TUint32*)&xRight)[0]=STATIC_CAST(const TUint32*,aRight)[1];
677 ((TUint32*)&xRight)[1]=STATIC_CAST(const TUint32*,aRight)[0];
678 rr=Comp::Compare(xLeft,xRight);
682 case EDbColLongText8:
683 rr=iTextOps->Compare(STATIC_CAST(const TUint8*,aLeft),size,STATIC_CAST(const TUint8*,aRight),size);
686 case EDbColLongText16:
687 rr=iTextOps->Order(STATIC_CAST(const TUint16*,aLeft),size>>1,STATIC_CAST(const TUint16*,aRight),size>>1);
691 return key->iOrder==TDbKeyCol::EAsc ? rr : -rr;
695 aLeft=PtrAdd(aLeft,size);
696 aRight=PtrAdd(aRight,size);
701 // No clever stuff yet
703 void CDbStoreIndex::HKey::Between(const TAny* aLeft,const TAny* /*aRight*/,TBtreePivot& aPivot) const
705 aPivot.Copy((const TUint8*)aLeft,KeySize());
708 // Class CDbStoreIndex::HDupKey
710 TInt CDbStoreIndex::HDupKey::KeySize() const
716 // Report 'true' if the lookup key is not the entire B+tree key
717 // For a duplicates index this is if there is a restriction (as record id is ingored)
719 TBool CDbStoreIndex::HDupKey::IncompleteKey() const
721 return Restriction()!=0;
725 // The key includes the record id
727 const TAny* CDbStoreIndex::HDupKey::Key(const TAny* aRecord) const
732 TInt CDbStoreIndex::HDupKey::Compare(const TAny* aLeft,const TAny* aRight) const
734 const TDbRecordId* const left=(const TDbRecordId*)aLeft;
735 const TDbRecordId* const right=(const TDbRecordId*)aRight;
736 TInt rr=HKey::Compare(left+1,right+1);
737 if (rr==0 && FullComparison())
738 return Comp::Compare(left->Value(),right->Value());
742 // Class CDbStoreIndex::CIter
744 NONSHARABLE_CLASS(CDbStoreIndex::CIter) : public CDbRecordIter
747 static CIter* NewL(CDbStoreIndex& aIndex,TUint aInclusion,const TDbLookupKey* aLowerBound,const TDbLookupKey* aUpperBound);
749 inline CIter(CDbStoreIndex& aIndex);
751 void ConstructL(TUint aInclusion,const TDbLookupKey* aLowerBound,const TDbLookupKey* aUpperBound);
752 TAny* BoundL(const TDbLookupKey& aBound,const TAny*& aRestriction);
754 inline const CDbStoreIndex& Index() const;
755 TBool FindL(TDbRecordId aRecordId,const RDbTableRow& aRow,TBtree::TFind aFind);
756 TInt CompareL(const TAny* aBound,const TAny* aRestriction);
757 TBool _GotoL(TDbPosition aPosition);
760 TDbRecordId CurrentL();
761 TBool GotoL(TDbPosition aPosition);
762 TBool GotoL(TDbRecordId aRecordId,RDbTableRow& aBuffer);
763 TBool SeekL(const TDbLookupKey& aKey,RDbTable::TComparison aComparison);
764 TDeleted DoDeletedL(TDbPosition aPosition,TDbRecordId aRecordId,const RDbTableRow* aRow);
772 const TAny* iLowerRestriction;
774 const TAny* iUpperRestriction;
777 inline CDbStoreIndex::CIter::CIter(CDbStoreIndex& aIndex) :
778 CDbRecordIter(aIndex)
782 inline const CDbStoreIndex& CDbStoreIndex::CIter::Index() const
784 return STATIC_CAST(CDbStoreIndex&,Host());
787 CDbStoreIndex::CIter* CDbStoreIndex::CIter::NewL(CDbStoreIndex& aIndex,TUint aInclusion,const TDbLookupKey* aLowerBound,const TDbLookupKey* aUpperBound)
789 CIter* self=new(ELeave) CIter(aIndex);
790 CleanupStack::PushL(self);
791 self->ConstructL(aInclusion,aLowerBound,aUpperBound);
796 CDbStoreIndex::CIter::~CIter()
798 User::Free(iLowerBound);
799 User::Free(iUpperBound);
802 void CDbStoreIndex::CIter::ConstructL(TUint aInclusion,const TDbLookupKey* aLowerBound,const TDbLookupKey* aUpperBound)
806 TBtree::TFind seek=TBtree::EGreaterEqual;
807 if ((aInclusion&CDbRecordIndex::EIncludeLower)==0)
809 seek=TBtree::EGreaterThan;
812 iLowerSeek=TUint8(seek);
813 iLowerBound=BoundL(*aLowerBound,iLowerRestriction);
817 TBtree::TFind seek=TBtree::ELessThan;
818 if (aInclusion&CDbRecordIndex::EIncludeUpper)
820 seek=TBtree::ELessEqual;
823 iUpperSeek=TUint8(seek);
824 iUpperBound=BoundL(*aUpperBound,iUpperRestriction);
829 // Construct and allocate a key for the boundary value
831 TAny* CDbStoreIndex::CIter::BoundL(const TDbLookupKey& aBound,const TAny*& aRestriction)
833 TUint8 entry[KMaxBtreeKeyLength];
834 HKey& key=Index().Key();
835 TInt size=key.EntryL(entry,aBound);
836 const TUint8* ekey=(const TUint8*)key.Key(entry);
838 TAny* e=User::AllocL(size);
839 Mem::Copy(e,ekey,size);
840 aRestriction=key.Restriction();
845 // Extract the current key and compare it with a boundary key
847 TInt CDbStoreIndex::CIter::CompareL(const TAny* aBound,const TAny* aRestriction)
849 TUint8 entry[KMaxBtreeKeyLength];
850 HKey& key=Index().Key();
851 key.Restrict(aRestriction);
852 Index().Tree().ExtractAtL(iPos,entry,key.EntrySize());
853 return key.Compare(key.Key(entry),aBound);
857 // return the cardinality of the index
859 TInt CDbStoreIndex::CIter::Count() const
862 return KDbUndefinedCount;
864 return KDbUndefinedCount;
865 return Index().Count();
869 // return the current record id
871 TDbRecordId CDbStoreIndex::CIter::CurrentL()
874 Index().Tree().ExtractAtL(iPos,&id,sizeof(id));
879 // iterate to the required position, does not test the boundary condition
881 TBool CDbStoreIndex::CIter::_GotoL(TDbPosition aPosition)
883 const TBtree& tree=Index().Tree();
886 default: // all control paths return a value
889 return tree.NextL(iPos);
891 return tree.PreviousL(iPos);
894 return tree.FirstL(iPos);
895 Index().Key().Restrict(iLowerRestriction);
896 return tree.FindL(iPos,iLowerBound,TBtree::TFind(iLowerSeek));
899 return tree.LastL(iPos);
900 Index().Key().Restrict(iUpperRestriction);
901 return tree.FindL(iPos,iUpperBound,TBtree::TFind(iUpperSeek));
906 // iterate to the required position and check that it is in bounds
908 TBool CDbStoreIndex::CIter::GotoL(TDbPosition aPosition)
910 TBool r=_GotoL(aPosition);
913 if (aPosition==EDbFirst || aPosition==EDbNext)
916 return CompareL(iUpperBound,iUpperRestriction)-iUpperCheck<0;
921 return CompareL(iLowerBound,iLowerRestriction)-iLowerCheck>=0;
928 // Construct the Btree key for the row and lookup
930 TBool CDbStoreIndex::CIter::FindL(TDbRecordId aRecordId,const RDbTableRow& aRow,TBtree::TFind aFind)
932 TUint8 entry[KMaxBtreeKeyLength];
933 HKey& key=Index().Key();
934 key.EntryL(entry,aRow,aRecordId);
935 return Index().Tree().FindL(iPos,key.Key(entry),aFind);
939 // Go directly to a row
941 TBool CDbStoreIndex::CIter::GotoL(TDbRecordId aRecordId,RDbTableRow& aRow)
943 aRow.ReadL(aRecordId);
944 return FindL(aRecordId,aRow,TBtree::EEqualTo);
948 // Do a keyed lookup in the index
950 TBool CDbStoreIndex::CIter::SeekL(const TDbLookupKey& aKey,RDbTable::TComparison aComparison)
952 TUint8 entry[KMaxBtreeKeyLength];
953 HKey& key=Index().Key();
954 key.EntryL(entry,aKey);
955 const TAny* ekey=key.Key(entry);
961 case RDbTable::ELessThan:
962 find=TBtree::ELessThan;
964 case RDbTable::ELessEqual:
965 find=TBtree::ELessEqual;
967 case RDbTable::EEqualTo:
968 if (key.IncompleteKey())
970 // The B+tree search code cannot correctly do a == search when the
971 // comparison key is not complete. Instead we do a >= search and then
972 // check the returned entry does match
974 if (!Index().Tree().FindL(iPos,ekey,TBtree::EGreaterEqual))
975 return EFalse; // off the end
976 return CompareL(ekey,key.Restriction())==0;
978 find=TBtree::EEqualTo;
980 case RDbTable::EGreaterEqual:
981 find=TBtree::EGreaterEqual;
983 case RDbTable::EGreaterThan:
984 find=TBtree::EGreaterThan;
987 return Index().Tree().FindL(iPos,ekey,find);
991 // Set the iterator following a record deletion
993 CDbStoreIndex::CIter::TDeleted CDbStoreIndex::CIter::DoDeletedL(TDbPosition aPosition,TDbRecordId aRecordId,const RDbTableRow* aRow)
996 return ENotSupported;
997 return FindL(aRecordId,*aRow,aPosition==EDbNext ? TBtree::EGreaterEqual : TBtree::ELessEqual) ? EAtRow : ENoRow;
1000 // Class CDbStoreIndex
1002 CDbStoreIndex::CDbStoreIndex(CDbStoreDatabase& aDatabase,const CDbStoreIndexDef& aDef) :
1003 iDatabase(aDatabase),
1005 iStats(MUTABLE_CAST(TDbStoreIndexStats&,aDef.iStats))
1009 CDbStoreIndex::~CDbStoreIndex()
1015 // Create the persistent representation of the index in the store
1017 TStreamId CDbStoreIndex::CreateL(CDbStoreDatabase& aDatabase,const CDbStoreIndexDef& aDef)
1019 MUTABLE_CAST(TDbStoreIndexStats&,aDef.iStats).Reset();
1020 RStoreWriteStream strm;
1021 TStreamId id=strm.CreateLC(aDatabase.Store());
1022 strm<<KEmptyBtreeToken<<aDef.iStats;
1024 CleanupStack::PopAndDestroy();
1029 // Check the index for damage (without constructing the object)
1031 TBool CDbStoreIndex::IsDamagedL(CDbStoreDatabase& aDatabase,const CDbStoreIndexDef& aDef)
1033 RStoreReadStream strm;
1034 strm.OpenLC(aDatabase.Store(),aDef.TokenId());
1037 CleanupStack::PopAndDestroy();
1038 return token.IsBroken();
1042 // Create a StoreIndex object
1044 CDbStoreIndex* CDbStoreIndex::NewL(CDbStoreDatabase& aDatabase,const CDbStoreIndexDef& aDef,const CDbTableDef& aTable)
1046 CDbStoreIndex* self=new(ELeave) CDbStoreIndex(aDatabase,aDef);
1047 CleanupStack::PushL(self);
1048 self->iTokenId=aDef.TokenId();
1049 HKey* key=self->iKey=HKey::NewL(aDef.Key(),aTable.Columns());
1050 self->iLeafOrg.SetEntrySize(key->EntrySize());
1051 self->iIndexOrg.SetEntrySize(key->KeySize());
1052 self->iTree.Connect(&aDatabase.PagePoolL(),key,&self->iLeafOrg,&self->iIndexOrg);
1053 CleanupStack::Pop();
1058 // restore from the Store
1060 TBool CDbStoreIndex::RestoreL()
1062 RStoreReadStream strm;
1063 strm.OpenLC(iDatabase.Store(),iTokenId);
1065 strm>>token>>iStats;
1066 CleanupStack::PopAndDestroy();
1067 iTree.Set(token,EBtreeFast);
1068 return iTree.IsBroken();
1072 // Update the index statistics from the index
1074 void CDbStoreIndex::RefreshStatsL()
1078 if (iTree.FirstL(pos))
1080 TUint8 entry[KMaxBtreeKeyLength];
1081 Tree().ExtractAtL(pos,entry,key.EntrySize());
1082 key.Bound(iStats.iLower,entry);
1084 Tree().ExtractAtL(pos,entry,key.EntrySize());
1085 key.Bound(iStats.iUpper,entry);
1087 iStats.Refresh(key.KeyType());
1091 // Framework member: synchronise the persistent data
1093 void CDbStoreIndex::SynchL()
1095 if (iStats.NeedsRefresh())
1097 RStoreWriteStream strm;
1098 strm.ReplaceLC(iDatabase.Store(),iTokenId);
1099 strm<<iTree.Token()<<iStats;
1101 CleanupStack::PopAndDestroy();
1105 // Framework member: mark the persistent token dirty
1107 void CDbStoreIndex::AboutToModifyL()
1110 if (!iTree.IsEmpty()) // empty btrees are unbreakable
1112 TBtreeToken token=iTree.Token();
1113 token.Touch(); // mark persistent data as broken
1114 RStoreWriteStream strm;
1115 strm.OpenLC(iDatabase.Store(),iTokenId);
1118 CleanupStack::PopAndDestroy();
1123 // Try to find a record in the index
1124 // return ENoMatch if no key match, EEntryMatch if entire entry is present
1125 // EKeyMatch if only key matches (only for unique indexes)
1127 CDbStoreIndex::TFind CDbStoreIndex::FindL(TDbRecordId aRecordId,const RDbTableRow& aRow)
1129 __ASSERT((!iTree.IsEmpty())==(Count()!=0));
1130 TUint8 entry[KMaxBtreeKeyLength];
1131 iKey->EntryL(entry,aRow,aRecordId);
1133 if (!iTree.FindL(pos,iKey->Key(entry)))
1136 iTree.ExtractAtL(pos,&id,sizeof(id));
1137 return id==aRecordId ? EEntryMatch : EKeyMatch;
1141 // Add the row to the index
1142 // return True if insertion was good, false if duplicate found
1144 TBool CDbStoreIndex::DoInsertL(TDbRecordId aRecordId,const RDbTableRow& aRow)
1146 __ASSERT((!iTree.IsEmpty())==(Count()!=0));
1147 TUint8 entry[KMaxBtreeKeyLength];
1148 TInt len=iKey->EntryL(entry,aRow,aRecordId);
1150 TBool insert=iTree.InsertL(pos,entry,len);
1157 // Remove row from index
1159 void CDbStoreIndex::DoDeleteL(TDbRecordId aRecordId,const RDbTableRow& aRow)
1161 __ASSERT((!iTree.IsEmpty())==(Count()!=0));
1162 TUint8 entry[KMaxBtreeKeyLength];
1163 iKey->EntryL(entry,aRow,aRecordId);
1164 __DEBUG(TInt dbgchk=) iTree.DeleteL(iKey->Key(entry));
1170 // Provide an iterator for the index ordering
1172 CDbRecordIter* CDbStoreIndex::IteratorL(TUint aInclusion,const TDbLookupKey* aLowerBound,const TDbLookupKey* aUpperBound)
1174 return CIter::NewL(*this,aInclusion,aLowerBound,aUpperBound);
1178 // repair the tree from the sequence set after reclamation of the page pool
1180 void CDbStoreIndex::RepairL()
1182 if (!iTree.IsEmpty()) // empty trees are unbreakable
1186 iStats.Reset(iTree.RepairL());
1192 // Throw away the index data, also used prior to a recovering rebuild
1194 void CDbStoreIndex::DiscardL()
1203 // Throw away the token
1205 void CDbStoreIndex::DestroyL()
1207 iDatabase.Store().DeleteL(iTokenId);
1210 // Class CDbStoreIndex::CDiscarder
1212 CDbStoreIndex::CDiscarder::CDiscarder()
1215 CDbStoreIndex::CDiscarder::~CDiscarder()
1220 TInt CDbStoreIndex::CDiscarder::Open(CDbStoreIndex* anIndex)
1228 // Do the single step of index discard
1230 TInt CDbStoreIndex::CDiscarder::StepL(TInt)
1233 CDbStoreIndex& index=*iIndex;