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.
14 // Order-by stage for the cursor data "pipeline"
22 #define _CHECK_ORDERING // forces a full test of the ordering after sorting
25 #define MAX( a, b ) ( (a) > (b) ? (a) : (b) )
29 Each key value is always aligned on a 32-bit boundary to allow word reads and writes
30 integral values are always atored as 32-bit values just as for the row buffer
31 Text columns are encoded as follows (trailing padding is omitted from the description)
33 Text8 columns <byte n><n ASCII characters>
34 n is the [character] length of the column
36 Text16 columns <short n><n UNICODE characters>
37 n is the [character] length of the column
39 LongText8 columns <word n><n ASCII characters>
40 or <word n|ETrunc><32 ASCII characters><blobId>
41 n is [byte] size of the column
43 LongText16 columns <word n><n/2 UNICODE characters>
44 or <word n|ETrunc><16 UNICODE characters><blobId>
45 n is [byte] size of the column
50 class CDbOrderByStage::HOrdering
62 enum {ETrunc=(TInt)0x80000000};
63 enum {ETruncSize=32}; // store 32 bytes of truncated BLOBs
65 inline TInt Size() const;
68 union {TUint8 iData8[ETruncSize]; TUint16 iData16[ETruncSize>>1];};
72 static HOrdering* NewL(CDbTable& aTable,const CDbKey& aKey);
73 TAny* EntryL(TAny* aPtr,const RDbTableRow& aRow) const;
74 TInt CompareL(const TAny* aLeft,const TAny* aRight) const;
77 inline HOrdering(TInt aCount,TDbTextComparison aComparison,CDbTable& aTable);
78 MStreamBuf* BlobLC(TDbBlobId aId,TDbColType aType) const;
79 TInt CompareLongText8L(const TBlobKey& aLeft,const TBlobKey& aRight) const;
80 TInt CompareLongText16L(const TBlobKey& aLeft,const TBlobKey& aRight) const;
83 const TTextOps& iTextOps;
84 const TKeyCol* iEndOfKeys;
85 TKeyCol iKeys[1]; // one or more keys
90 inline TInt CDbOrderByStage::HOrdering::TBlobKey::Size() const
92 TInt size=_FOFF(TBlobKey,iData8[iSize+3]);
93 return (size&ETrunc) ? sizeof(TBlobKey) : size&~3;
96 inline CDbOrderByStage::HOrdering::HOrdering(TInt aCount,TDbTextComparison aComparison,CDbTable& aTable) :
98 iTextOps(TTextOps::Ops(aComparison)),
99 iEndOfKeys(&iKeys[aCount])
104 // Construct the ordering based on the key definition (must be valid for the table)
105 // and the column set provided
107 CDbOrderByStage::HOrdering* CDbOrderByStage::HOrdering::NewL(CDbTable& aTable,const CDbKey& aKey)
109 TInt count=aKey.Count();
111 HOrdering* self=new(User::AllocLC(_FOFF(HOrdering,iKeys[count])))
112 HOrdering(count,aKey.Comparison(),aTable);
113 TKeyCol* pKey=&self->iKeys[0];
114 const HDbColumnSet& columns=aTable.Def().Columns();
115 for (TInt ii=0;ii<count;++pKey,++ii)
117 const TDbKeyCol& key=aKey[ii];
118 pKey->iOrder=TUint8(key.iOrder);
119 pKey->iOrdinal=columns.ColNoL(key.iName);
120 if (pKey->iOrdinal==KDbNullColNo)
121 __LEAVE(KErrNotFound);
122 const TDbColumnDef& col=columns[pKey->iOrdinal];
123 switch (pKey->iType=col.iType)
130 TInt l=col.iMaxLength;
133 pKey->iLength=TUint8(l);
137 // fall through to argument error
139 case EDbColLongBinary:
140 __LEAVE(KErrArgument);
147 TInt CDbOrderByStage::HOrdering::MaxSize() const
150 const TKeyCol* const end=iEndOfKeys;
151 const TKeyCol* key=&iKeys[0];
170 s=Align4(key->iLength+1);
173 s=Align4((key->iLength<<1)+1);
175 case EDbColLongText8:
176 case EDbColLongText16:
177 s=MAX(__Align8(sizeof(TUint32)+KDbMaxInlineBlobSize),sizeof(TBlobKey));
185 MStreamBuf* CDbOrderByStage::HOrdering::BlobLC(TDbBlobId aId,TDbColType aType) const
187 return iTable.BlobsL()->ReadLC(aId,aType);
191 // Construct an entry at aPtr from the record given
192 // iMaxSize bytes are assumed to be available at aPtr
193 // return the actual size of the entry
195 TAny* CDbOrderByStage::HOrdering::EntryL(TAny* aPtr,const RDbTableRow& aRow) const
197 __ASSERT(Align4(aPtr)==aPtr);
198 TUint32* ptr=(TUint32*)aPtr; // 32-bit words are nice
199 const TKeyCol* const end=iEndOfKeys;
200 const TKeyCol* key=iKeys;
203 const TDbCell* cell=aRow.ColCell(key->iOrdinal);
204 TDbColType type=TDbColType(key->iType);
205 if (cell->Length()==0)
207 const TUint K64BitColumnMask=(1u<<EDbColInt64)|(1u<<EDbColReal64)|(1u<<EDbColDateTime);
209 if (K64BitColumnMask&(1u<<type))
210 *ptr++=0; // 8-byte column
225 *ptr++=*(const TUint32*)cell->Data();
231 const TUint32* data=(const TUint32*)cell->Data();
238 TInt size=cell->Length();
239 *(TUint8*)ptr=TUint8(size);
240 ptr=(TUint32*)Align4(Mem::Copy(PtrAdd(ptr,1),cell->Data(),size));
245 TInt size=cell->Length();
246 *(TUint16*)ptr=TUint16(size>>1);
247 ptr=(TUint32*)Align4(Mem::Copy(PtrAdd(ptr,2),cell->Data(),size));
250 case EDbColLongText8:
251 case EDbColLongText16:
253 const TDbBlob& blob=*(const TDbBlob*)cell->Data();
254 TInt size=blob.Size();
258 ptr=(TUint32*)Align4(Mem::Copy(ptr,blob.Data(),size));
263 if (size>TBlobKey::ETruncSize)
265 size|=TBlobKey::ETrunc;
266 rsize=TBlobKey::ETruncSize;
269 BlobLC(blob.Id(),TDbColType(key->iType))->ReadL(ptr,rsize);
270 CleanupStack::PopAndDestroy();
271 ptr=Align4(PtrAdd(ptr,rsize));
272 if (size&TBlobKey::ETrunc)
282 TInt CDbOrderByStage::HOrdering::CompareLongText8L( const CDbOrderByStage::HOrdering::TBlobKey& aLeft, const CDbOrderByStage::HOrdering::TBlobKey& aRight ) const
284 TUint sizel = aLeft.iSize;
285 TUint sizer = aRight.iSize;
289 if ( s > static_cast<TUint>( TBlobKey::ETruncSize ) && ( ( sizel | sizer ) & static_cast<TUint>( TBlobKey::ETrunc ) ) )
290 s = TBlobKey::ETruncSize;
291 TInt rr = iTextOps.Compare( aLeft.iData8, s, aRight.iData8, s );
295 // matches up to the same-length inlined data...
296 // now it gets interesting
298 if ( ( ( sizel | sizer ) & static_cast<TUint>( TBlobKey::ETrunc ) ) == 0 )
299 return sizel - sizer; // neither are truncated
301 return -1; // left completely match against truncated right
303 return 1; // right completely matched against truncated left
305 s = Min( TInt( sizel & ~TBlobKey::ETrunc ), TInt( sizer & ~TBlobKey::ETrunc ) ); // minimum real length
306 if ( sizel & static_cast<TUint>( TBlobKey::ETrunc ) )
308 MStreamBuf& bufL = *BlobLC( aLeft.iId, EDbColLongText8 );
309 if ( sizer & static_cast<TUint>( TBlobKey::ETrunc ) )
310 { // both out-of-line
311 rr = Comp::Compare8L( bufL, *BlobLC( aRight.iId, EDbColLongText8 ), s, iTextOps );
312 CleanupStack::PopAndDestroy();
314 else // left side only out-of-line
315 rr = Comp::Compare8L( bufL, aRight.iData8, s, iTextOps );
318 { // right side only out-of-line
319 __ASSERT( sizer & static_cast<TUint>( TBlobKey::ETrunc ) );
320 rr = -Comp::Compare8L( *BlobLC( aRight.iId, EDbColLongText8 ), aLeft.iData8, s, iTextOps );
322 CleanupStack::PopAndDestroy();
323 return rr ? rr : ( sizel & ~TBlobKey::ETrunc ) - ( sizer & ~TBlobKey::ETrunc );
326 TInt CDbOrderByStage::HOrdering::CompareLongText16L( const CDbOrderByStage::HOrdering::TBlobKey& aLeft, const CDbOrderByStage::HOrdering::TBlobKey& aRight ) const
328 TUint sizeL = aLeft.iSize & ~TBlobKey::ETrunc; // set truncation bit to 0 to get true size
329 TUint sizeR = aRight.iSize & ~TBlobKey::ETrunc;
330 TBool truncL = aLeft.iSize & TBlobKey::ETrunc; // data is truncated if TBlobKey::ETrunc bit is 1
331 TBool truncR = aRight.iSize & TBlobKey::ETrunc;
333 if (!(truncL | truncR)) // neither side is truncated, compare full strings
335 return iTextOps.Order( aLeft.iData16, sizeL>>1, aRight.iData16, sizeR>>1 );
339 TUint sizeMin = Min( TInt(sizeL), TInt(sizeR) ); // minimum real length
342 MStreamBuf& bufL = *BlobLC( aLeft.iId, EDbColLongText16 );
344 { // both out-of-line
345 result = Comp::Compare16L( bufL, *BlobLC( aRight.iId, EDbColLongText16 ), sizeMin, iTextOps );
346 CleanupStack::PopAndDestroy();
348 else // left side only out-of-line
349 result = Comp::Compare16L( bufL, aRight.iData16, sizeMin, iTextOps );
352 { // right side only out-of-line
354 result = -Comp::Compare16L( *BlobLC( aRight.iId, EDbColLongText16 ), aLeft.iData16, sizeMin, iTextOps );
356 CleanupStack::PopAndDestroy();
357 return result ? result : ( sizeL ) - ( sizeR );
360 TInt CDbOrderByStage::HOrdering::CompareL(const TAny* aLeft,const TAny* aRight) const
365 const TUint8* left=(const TUint8*)aLeft;
366 const TUint8* right=(const TUint8*)aRight;
367 const TKeyCol* end=iEndOfKeys;
368 const TKeyCol* key=&iKeys[0];
380 rr=Comp::Compare(*(const TUint32*)left,*(const TUint32*)right);
381 left+=sizeof(TUint32);
382 right+=sizeof(TUint32);
387 rr=Comp::Compare(*(const TInt32*)left,*(const TInt32*)right);
388 left+=sizeof(TInt32);
389 right+=sizeof(TInt32);
392 rr=Comp::Compare(*(const TInt64*)left,*(const TInt64*)right);
393 left+=sizeof(TInt64);
394 right+=sizeof(TInt64);
397 rr=Comp::Compare(*(const TTime*)left,*(const TTime*)right);
399 right+=sizeof(TTime);
402 rr=Comp::Compare(*(const TReal32*)left,*(const TReal32*)right);
403 left+=sizeof(TReal32);
404 right+=sizeof(TReal32);
407 rr=Comp::Compare(*(const TReal64*)left,*(const TReal64*)right);
408 left+=sizeof(TReal64);
409 right+=sizeof(TReal64);
415 rr=iTextOps.Compare(left,sizel,right,sizer);
416 left=Align4(left+sizel);
417 right=Align4(right+sizer);
422 const TUint16* l16=reinterpret_cast<const TUint16*>(left);
423 const TUint16* r16=reinterpret_cast<const TUint16*>(right);
426 rr=iTextOps.Order(l16,sizel,r16,sizer);
427 left=Align4(reinterpret_cast<const TUint8*>(l16+sizel));
428 right=Align4(reinterpret_cast<const TUint8*>(r16+sizer));
431 case EDbColLongText8:
433 const TBlobKey* ltext=(const TBlobKey*)left;
434 const TBlobKey* rtext=(const TBlobKey*)right;
435 rr=CompareLongText8L(*ltext,*rtext);
437 right+=rtext->Size();
440 case EDbColLongText16:
442 const TBlobKey* ltext=(const TBlobKey*)left;
443 const TBlobKey* rtext=(const TBlobKey*)right;
444 rr=CompareLongText16L(*ltext,*rtext);
446 right+=rtext->Size();
455 return key->iOrder==TDbKeyCol::EAsc ? rr : -rr;
460 NONSHARABLE_CLASS(CDbOrderByStage::CKeys) : public CBase
462 public: // should be private but VC++ 4.0 whinges
466 TUint8 iKey[4]; // and then some
469 enum {EKeysGranularity=32};
475 enum {EMinPageSize=0x200,EMinKeyElements=2};
477 CKeys(TInt aMaxKeySize);
480 void AddL(TDbRecordId aRecordId,const RDbTableRow& aRow,const HOrdering& aOrder);
481 void SortL(const HOrdering& aOrder);
482 void GetRecordsL(CArrayFix<TDbRecordId>& aRecords);
483 #ifdef _CHECK_ORDERING
484 void VerifyOrderingL(const HOrdering& aOrder);
487 void SortL(TKey* aData[],TInt aElem,const HOrdering& aOrder);
489 void KeyComplete(TAny* aEnd);
492 RPointerArray<TKey> iKeys;
500 // Class CDbOrderByStage::CKeys
502 CDbOrderByStage::CKeys::CKeys(TInt aMaxKeySize)
503 :iKeys(EKeysGranularity),iMaxKeySize(_FOFF(TKey,iKey[aMaxKeySize])),
504 iPageSize(Max(EMinPageSize+iMaxKeySize,iMaxKeySize*EMinKeyElements))
507 CDbOrderByStage::CKeys::~CKeys()
513 void CDbOrderByStage::CKeys::SortL(CDbOrderByStage::CKeys::TKey* aData[],TInt aElem,const HOrdering& aOrder)
515 // Sort the array of row keys
530 aData[aElem]=aData[0];
539 for (parent=ix;;parent=ix)
544 if (ix<aElem && aOrder.CompareL(aData[ix]->iKey,aData[ix+1]->iKey)<0)
549 if (aOrder.CompareL(aData[ix]->iKey,key->iKey)<=0)
551 aData[parent]=aData[ix];
557 void CDbOrderByStage::CKeys::AddL(TDbRecordId aRecordId,const RDbTableRow& aRow,const HOrdering& aOrder)
559 // Create a key for the row and store it
563 key.iId=aRecordId.Value();
564 TAny* end=aOrder.EntryL(&key.iKey[0],aRow);
565 __LEAVE_IF_ERROR(iKeys.Append(&key));
569 void CDbOrderByStage::CKeys::SortL(const HOrdering& aOrder)
571 TInt elem=iKeys.Count();
573 SortL(&iKeys[0],elem,aOrder);
576 void CDbOrderByStage::CKeys::GetRecordsL(CArrayFix<TDbRecordId>& aRecords)
578 TInt elem=iKeys.Count();
581 TKey** const base=&iKeys[0];
582 TKey** ptr=base+elem;
585 *REINTERPRET_CAST(TDbRecordId*,ptr)=(*ptr)->iId;
587 Release(); // discard key memory before adding records
588 aRecords.InsertL(0,REINTERPRET_CAST(const TDbRecordId*,base),elem);
592 void CDbOrderByStage::CKeys::Release()
594 // Discard all of the allocated pages
608 CDbOrderByStage::CKeys::TKey& CDbOrderByStage::CKeys::NewKeyL()
610 // Allocate a key entry for the raw key data
614 if (PtrAdd(p,iMaxKeySize)>iEnd)
615 { // not enough space for a maximum key
618 { // compress the current page
619 __DEBUG(page=(TPage*))User::ReAlloc(page,(TUint8*)iNext-(TUint8*)page);
620 __ASSERT(page==iPages); // if it moves we are dead
622 page=(TPage*)User::AllocL(_FOFF(TPage,iEntries)+iPageSize);
625 p=iNext=&page->iEntries[0];
626 iEnd=PtrAdd(p,iPageSize);
631 void CDbOrderByStage::CKeys::KeyComplete(TAny* aEnd)
633 // Key is complete, prepare for the next one
636 __ASSERT(aEnd==Align4(aEnd));
637 __ASSERT(iNext<=aEnd&&aEnd<=iEnd);
638 iNext=STATIC_CAST(TKey*,aEnd);
641 #ifdef _CHECK_ORDERING
642 void CDbOrderByStage::CKeys::VerifyOrderingL(const HOrdering& aOrder)
644 // this performs a full O(N*N) comparison of the record set
645 if (iKeys.Count()==0)
647 TKey* const* data=&iKeys[0];
648 for (TInt ii=iKeys.Count();--ii>=0;)
650 for (TInt jj=iKeys.Count();--jj>=0;)
652 TInt rr=aOrder.CompareL(data[ii]->iKey,data[jj]->iKey);
654 __ASSERT_ALWAYS(rr==0,User::Invariant());
656 __ASSERT_ALWAYS(rr<=0,User::Invariant());
658 __ASSERT_ALWAYS(rr>=0,User::Invariant());
664 // Class CDbOrderByStage
666 inline const RDbTableRow& CDbOrderByStage::Row()
668 inline CDbOrderByStage::CKeys& CDbOrderByStage::Keys()
669 {__ASSERT(iKeys);return *iKeys;}
670 inline const CDbOrderByStage::HOrdering& CDbOrderByStage::Order()
671 {__ASSERT(iOrder);return *iOrder;}
673 CDbOrderByStage::CDbOrderByStage(const RDbTableRow& aRow)
674 : CDbBasicWindowStage(KDbUnlimitedWindow),iRow(aRow)
676 __ASSERT(iState==EReset);
679 CDbOrderByStage::~CDbOrderByStage()
685 void CDbOrderByStage::ConstructL(const CDbKey& aOrderBy)
687 // Build the key structures to support the partial and complete ordering
690 iOrder=HOrdering::NewL(Row().Table(),aOrderBy);
693 void CDbOrderByStage::Reset()
695 // Reset the window to initial state
701 CDbBasicWindowStage::Reset();
704 TBool CDbOrderByStage::ReadL(TInt& aWork,TDbPosition aNext)
706 // Read some more record keys
709 TDbRecordId id(KDbNullRecordIdValue);
711 while ((r=CDbDataStage::GotoL(aWork,aNext,id))==ESuccess)
713 CDbDataStage::ReadRowL(id);
714 Keys().AddL(id,Row(),Order());
722 __LEAVE(KErrNotReady);
730 TBool CDbOrderByStage::DoEvaluateL(TInt& aWork)
732 // Build the key collection, and finally sort it
742 __LEAVE(KErrNotReady);
745 iKeys=new(ELeave) CKeys(Order().MaxSize());
746 // drop through to EEvaluate
748 if (ReadL(aWork,s==EReset ? EDbFirst : EDbNext))
753 // drop through to ESort
755 Keys().SortL(Order());
756 #ifdef _CHECK_ORDERING
757 Keys().VerifyOrderingL(Order());
759 // drop through to EAdd
761 Keys().GetRecordsL(iRecords);
764 // drop through to EComplete
771 TBool CDbOrderByStage::Unevaluated()
773 // Return whether it is worth Evaluating
776 if (iState==EComplete)
777 return CDbDataStage::Unevaluated();
778 return iState!=EFailed;
782 // Class CDbReorderWindowStage
784 CDbReorderWindowStage::CDbReorderWindowStage()
785 : CDbBasicWindowStage(KDbUnlimitedWindow),iRows(EGranularity)
787 __ASSERT(iState==EReset);
790 CDbReorderWindowStage::~CDbReorderWindowStage()
795 void CDbReorderWindowStage::Reset()
797 // Reset the window to initial state
802 CDbBasicWindowStage::Reset();
805 TBool CDbReorderWindowStage::ReadL(TInt& aWork,TDbPosition aNext)
807 // Read some more row ids
810 TDbRecordId id(KDbNullRecordIdValue);
812 while ((r=CDbDataStage::GotoL(aWork,aNext,id))==ESuccess)
814 __LEAVE_IF_ERROR(iRows.Append(id.Value()));
822 __LEAVE(KErrNotReady);
830 TBool CDbReorderWindowStage::DoEvaluateL(TInt& aWork)
832 // Build the key collection, and finally sort it
842 __LEAVE(KErrNotReady);
845 if (ReadL(aWork,s==EReset ? EDbFirst : EDbNext))
853 iRecords.AppendL((const TDbRecordId*)&iRows[0],iRows.Count());
862 TBool CDbReorderWindowStage::Unevaluated()
864 // Return whether it is worth Evaluating
867 if (iState==EComplete)
868 return CDbDataStage::Unevaluated();
869 return iState!=EFailed;