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"
20 #include <d32dbmsconstants.h>
23 #define _CHECK_ORDERING // forces a full test of the ordering after sorting
26 #define MAX( a, b ) ( (a) > (b) ? (a) : (b) )
30 Each key value is always aligned on a 32-bit boundary to allow word reads and writes
31 integral values are always atored as 32-bit values just as for the row buffer
32 Text columns are encoded as follows (trailing padding is omitted from the description)
34 Text8 columns <byte n><n ASCII characters>
35 n is the [character] length of the column
37 Text16 columns <short n><n UNICODE characters>
38 n is the [character] length of the column
40 LongText8 columns <word n><n ASCII characters>
41 or <word n|ETrunc><32 ASCII characters><blobId>
42 n is [byte] size of the column
44 LongText16 columns <word n><n/2 UNICODE characters>
45 or <word n|ETrunc><16 UNICODE characters><blobId>
46 n is [byte] size of the column
51 class CDbOrderByStage::HOrdering
63 enum {ETrunc=(TInt)0x80000000};
64 enum {ETruncSize=32}; // store 32 bytes of truncated BLOBs
66 inline TInt Size() const;
69 union {TUint8 iData8[ETruncSize]; TUint16 iData16[ETruncSize>>1];};
73 static HOrdering* NewL(CDbTable& aTable,const CDbKey& aKey);
74 TAny* EntryL(TAny* aPtr,const RDbTableRow& aRow) const;
75 TInt CompareL(const TAny* aLeft,const TAny* aRight) const;
78 inline HOrdering(TInt aCount,TDbTextComparison aComparison,CDbTable& aTable);
79 MStreamBuf* BlobLC(TDbBlobId aId,TDbColType aType) const;
80 TInt CompareLongText8L(const TBlobKey& aLeft,const TBlobKey& aRight) const;
81 TInt CompareLongText16L(const TBlobKey& aLeft,const TBlobKey& aRight) const;
84 const TTextOps& iTextOps;
85 const TKeyCol* iEndOfKeys;
86 TKeyCol iKeys[1]; // one or more keys
91 inline TInt CDbOrderByStage::HOrdering::TBlobKey::Size() const
93 TInt size=_FOFF(TBlobKey,iData8[iSize+3]);
94 return (size&ETrunc) ? sizeof(TBlobKey) : size&~3;
97 inline CDbOrderByStage::HOrdering::HOrdering(TInt aCount,TDbTextComparison aComparison,CDbTable& aTable) :
99 iTextOps(TTextOps::Ops(aComparison)),
100 iEndOfKeys(&iKeys[aCount])
105 // Construct the ordering based on the key definition (must be valid for the table)
106 // and the column set provided
108 CDbOrderByStage::HOrdering* CDbOrderByStage::HOrdering::NewL(CDbTable& aTable,const CDbKey& aKey)
110 TInt count=aKey.Count();
112 HOrdering* self=new(User::AllocLC(_FOFF(HOrdering,iKeys[count])))
113 HOrdering(count,aKey.Comparison(),aTable);
114 TKeyCol* pKey=&self->iKeys[0];
115 const HDbColumnSet& columns=aTable.Def().Columns();
116 for (TInt ii=0;ii<count;++pKey,++ii)
118 const TDbKeyCol& key=aKey[ii];
119 pKey->iOrder=TUint8(key.iOrder);
120 pKey->iOrdinal=columns.ColNoL(key.iName);
121 if (pKey->iOrdinal==KDbNullColNo)
122 __LEAVE(KErrNotFound);
123 const TDbColumnDef& col=columns[pKey->iOrdinal];
124 switch (pKey->iType=col.iType)
131 TInt l=col.iMaxLength;
134 pKey->iLength=TUint8(l);
138 // fall through to argument error
140 case EDbColLongBinary:
141 __LEAVE(KErrArgument);
148 TInt CDbOrderByStage::HOrdering::MaxSize() const
151 const TKeyCol* const end=iEndOfKeys;
152 const TKeyCol* key=&iKeys[0];
171 s=Align4(key->iLength+1);
174 s=Align4((key->iLength<<1)+1);
176 case EDbColLongText8:
177 case EDbColLongText16:
178 s=MAX(__Align8(sizeof(TUint32)+KDbMaxInlineBlobSize),sizeof(TBlobKey));
186 MStreamBuf* CDbOrderByStage::HOrdering::BlobLC(TDbBlobId aId,TDbColType aType) const
188 return iTable.BlobsL()->ReadLC(aId,aType);
192 // Construct an entry at aPtr from the record given
193 // iMaxSize bytes are assumed to be available at aPtr
194 // return the actual size of the entry
196 TAny* CDbOrderByStage::HOrdering::EntryL(TAny* aPtr,const RDbTableRow& aRow) const
198 __ASSERT(Align4(aPtr)==aPtr);
199 TUint32* ptr=(TUint32*)aPtr; // 32-bit words are nice
200 const TKeyCol* const end=iEndOfKeys;
201 const TKeyCol* key=iKeys;
204 const TDbCell* cell=aRow.ColCell(key->iOrdinal);
205 TDbColType type=TDbColType(key->iType);
206 if (cell->Length()==0)
208 const TUint K64BitColumnMask=(1u<<EDbColInt64)|(1u<<EDbColReal64)|(1u<<EDbColDateTime);
210 if (K64BitColumnMask&(1u<<type))
211 *ptr++=0; // 8-byte column
226 *ptr++=*(const TUint32*)cell->Data();
232 const TUint32* data=(const TUint32*)cell->Data();
239 TInt size=cell->Length();
240 *(TUint8*)ptr=TUint8(size);
241 ptr=(TUint32*)Align4(Mem::Copy(PtrAdd(ptr,1),cell->Data(),size));
246 TInt size=cell->Length();
247 *(TUint16*)ptr=TUint16(size>>1);
248 ptr=(TUint32*)Align4(Mem::Copy(PtrAdd(ptr,2),cell->Data(),size));
251 case EDbColLongText8:
252 case EDbColLongText16:
254 const TDbBlob& blob=*(const TDbBlob*)cell->Data();
255 TInt size=blob.Size();
259 ptr=(TUint32*)Align4(Mem::Copy(ptr,blob.Data(),size));
264 if (size>TBlobKey::ETruncSize)
266 size|=TBlobKey::ETrunc;
267 rsize=TBlobKey::ETruncSize;
270 BlobLC(blob.Id(),TDbColType(key->iType))->ReadL(ptr,rsize);
271 CleanupStack::PopAndDestroy();
272 ptr=Align4(PtrAdd(ptr,rsize));
273 if (size&TBlobKey::ETrunc)
283 TInt CDbOrderByStage::HOrdering::CompareLongText8L( const CDbOrderByStage::HOrdering::TBlobKey& aLeft, const CDbOrderByStage::HOrdering::TBlobKey& aRight ) const
285 TUint sizel = aLeft.iSize;
286 TUint sizer = aRight.iSize;
290 if ( s > static_cast<TUint>( TBlobKey::ETruncSize ) && ( ( sizel | sizer ) & static_cast<TUint>( TBlobKey::ETrunc ) ) )
291 s = TBlobKey::ETruncSize;
292 TInt rr = iTextOps.Compare( aLeft.iData8, s, aRight.iData8, s );
296 // matches up to the same-length inlined data...
297 // now it gets interesting
299 if ( ( ( sizel | sizer ) & static_cast<TUint>( TBlobKey::ETrunc ) ) == 0 )
300 return sizel - sizer; // neither are truncated
302 return -1; // left completely match against truncated right
304 return 1; // right completely matched against truncated left
306 s = Min( TInt( sizel & ~TBlobKey::ETrunc ), TInt( sizer & ~TBlobKey::ETrunc ) ); // minimum real length
307 if ( sizel & static_cast<TUint>( TBlobKey::ETrunc ) )
309 MStreamBuf& bufL = *BlobLC( aLeft.iId, EDbColLongText8 );
310 if ( sizer & static_cast<TUint>( TBlobKey::ETrunc ) )
311 { // both out-of-line
312 rr = Comp::Compare8L( bufL, *BlobLC( aRight.iId, EDbColLongText8 ), s, iTextOps );
313 CleanupStack::PopAndDestroy();
315 else // left side only out-of-line
316 rr = Comp::Compare8L( bufL, aRight.iData8, s, iTextOps );
319 { // right side only out-of-line
320 __ASSERT( sizer & static_cast<TUint>( TBlobKey::ETrunc ) );
321 rr = -Comp::Compare8L( *BlobLC( aRight.iId, EDbColLongText8 ), aLeft.iData8, s, iTextOps );
323 CleanupStack::PopAndDestroy();
324 return rr ? rr : ( sizel & ~TBlobKey::ETrunc ) - ( sizer & ~TBlobKey::ETrunc );
327 TInt CDbOrderByStage::HOrdering::CompareLongText16L( const CDbOrderByStage::HOrdering::TBlobKey& aLeft, const CDbOrderByStage::HOrdering::TBlobKey& aRight ) const
329 TUint sizeL = aLeft.iSize & ~TBlobKey::ETrunc; // set truncation bit to 0 to get true size
330 TUint sizeR = aRight.iSize & ~TBlobKey::ETrunc;
331 TBool truncL = aLeft.iSize & TBlobKey::ETrunc; // data is truncated if TBlobKey::ETrunc bit is 1
332 TBool truncR = aRight.iSize & TBlobKey::ETrunc;
334 if (!(truncL | truncR)) // neither side is truncated, compare full strings
336 return iTextOps.Order( aLeft.iData16, sizeL>>1, aRight.iData16, sizeR>>1 );
340 TUint sizeMin = Min( TInt(sizeL), TInt(sizeR) ); // minimum real length
343 MStreamBuf& bufL = *BlobLC( aLeft.iId, EDbColLongText16 );
345 { // both out-of-line
346 result = Comp::Compare16L( bufL, *BlobLC( aRight.iId, EDbColLongText16 ), sizeMin, iTextOps );
347 CleanupStack::PopAndDestroy();
349 else // left side only out-of-line
350 result = Comp::Compare16L( bufL, aRight.iData16, sizeMin, iTextOps );
353 { // right side only out-of-line
355 result = -Comp::Compare16L( *BlobLC( aRight.iId, EDbColLongText16 ), aLeft.iData16, sizeMin, iTextOps );
357 CleanupStack::PopAndDestroy();
358 return result ? result : ( sizeL ) - ( sizeR );
361 TInt CDbOrderByStage::HOrdering::CompareL(const TAny* aLeft,const TAny* aRight) const
367 __ASSERT(aLeft != NULL);
368 __ASSERT(aRight != NULL);
370 const TUint8* left=(const TUint8*)aLeft;
371 const TUint8* right=(const TUint8*)aRight;
372 const TKeyCol* end=iEndOfKeys;
373 const TKeyCol* key=&iKeys[0];
385 rr=Comp::Compare(*(const TUint32*)left,*(const TUint32*)right);
386 left+=sizeof(TUint32);
387 right+=sizeof(TUint32);
392 rr=Comp::Compare(*(const TInt32*)left,*(const TInt32*)right);
393 left+=sizeof(TInt32);
394 right+=sizeof(TInt32);
397 rr=Comp::Compare(*(const TInt64*)left,*(const TInt64*)right);
398 left+=sizeof(TInt64);
399 right+=sizeof(TInt64);
402 rr=Comp::Compare(*(const TTime*)left,*(const TTime*)right);
404 right+=sizeof(TTime);
407 rr=Comp::Compare(*(const TReal32*)left,*(const TReal32*)right);
408 left+=sizeof(TReal32);
409 right+=sizeof(TReal32);
412 rr=Comp::Compare(*(const TReal64*)left,*(const TReal64*)right);
413 left+=sizeof(TReal64);
414 right+=sizeof(TReal64);
420 rr=iTextOps.Compare(left,sizel,right,sizer);
421 left=Align4(left+sizel);
422 right=Align4(right+sizer);
427 const TUint16* l16=reinterpret_cast<const TUint16*>(left);
428 const TUint16* r16=reinterpret_cast<const TUint16*>(right);
431 rr=iTextOps.Order(l16,sizel,r16,sizer);
432 left=Align4(reinterpret_cast<const TUint8*>(l16+sizel));
433 right=Align4(reinterpret_cast<const TUint8*>(r16+sizer));
436 case EDbColLongText8:
438 const TBlobKey* ltext=(const TBlobKey*)left;
439 const TBlobKey* rtext=(const TBlobKey*)right;
440 rr=CompareLongText8L(*ltext,*rtext);
442 right+=rtext->Size();
445 case EDbColLongText16:
447 const TBlobKey* ltext=(const TBlobKey*)left;
448 const TBlobKey* rtext=(const TBlobKey*)right;
449 rr=CompareLongText16L(*ltext,*rtext);
451 right+=rtext->Size();
460 return key->iOrder==TDbKeyCol::EAsc ? rr : -rr;
465 NONSHARABLE_CLASS(CDbOrderByStage::CKeys) : public CBase
467 public: // should be private but VC++ 4.0 whinges
471 TUint8 iKey[4]; // and then some
474 enum {EKeysGranularity=32};
480 enum {EMinPageSize=0x200,EMinKeyElements=2};
482 CKeys(TInt aMaxKeySize);
485 void AddL(TDbRecordId aRecordId,const RDbTableRow& aRow,const HOrdering& aOrder);
486 void SortL(const HOrdering& aOrder);
487 void GetRecordsL(CArrayFix<TDbRecordId>& aRecords);
488 #ifdef _CHECK_ORDERING
489 void VerifyOrderingL(const HOrdering& aOrder);
492 void SortL(TKey* aData[],TInt aElem,const HOrdering& aOrder);
494 void KeyComplete(TAny* aEnd);
497 RPointerArray<TKey> iKeys;
505 // Class CDbOrderByStage::CKeys
507 CDbOrderByStage::CKeys::CKeys(TInt aMaxKeySize)
508 :iKeys(EKeysGranularity),iMaxKeySize(_FOFF(TKey,iKey[aMaxKeySize])),
509 iPageSize(Max(EMinPageSize+iMaxKeySize,iMaxKeySize*EMinKeyElements))
512 CDbOrderByStage::CKeys::~CKeys()
518 void CDbOrderByStage::CKeys::SortL(CDbOrderByStage::CKeys::TKey* aData[],TInt aElem,const HOrdering& aOrder)
520 // Sort the array of row keys
535 aData[aElem]=aData[0];
544 for (parent=ix;;parent=ix)
549 if (ix<aElem && aOrder.CompareL(aData[ix]->iKey,aData[ix+1]->iKey)<0)
554 if (aOrder.CompareL(aData[ix]->iKey,key->iKey)<=0)
556 aData[parent]=aData[ix];
562 void CDbOrderByStage::CKeys::AddL(TDbRecordId aRecordId,const RDbTableRow& aRow,const HOrdering& aOrder)
564 // Create a key for the row and store it
568 key.iId=aRecordId.Value();
569 TAny* end=aOrder.EntryL(&key.iKey[0],aRow);
570 __LEAVE_IF_ERROR(iKeys.Append(&key));
574 void CDbOrderByStage::CKeys::SortL(const HOrdering& aOrder)
576 TInt elem=iKeys.Count();
578 SortL(&iKeys[0],elem,aOrder);
581 void CDbOrderByStage::CKeys::GetRecordsL(CArrayFix<TDbRecordId>& aRecords)
583 TInt elem=iKeys.Count();
586 TKey** const base=&iKeys[0];
587 TKey** ptr=base+elem;
590 *REINTERPRET_CAST(TDbRecordId*,ptr)=(*ptr)->iId;
592 Release(); // discard key memory before adding records
593 aRecords.InsertL(0,REINTERPRET_CAST(const TDbRecordId*,base),elem);
597 void CDbOrderByStage::CKeys::Release()
599 // Discard all of the allocated pages
613 CDbOrderByStage::CKeys::TKey& CDbOrderByStage::CKeys::NewKeyL()
615 // Allocate a key entry for the raw key data
619 if (PtrAdd(p,iMaxKeySize)>iEnd)
620 { // not enough space for a maximum key
623 { // compress the current page
624 __DEBUG(page=(TPage*))User::ReAlloc(page,(TUint8*)iNext-(TUint8*)page);
625 __ASSERT(page==iPages); // if it moves we are dead
627 page=(TPage*)User::AllocL(_FOFF(TPage,iEntries)+iPageSize);
630 p=iNext=&page->iEntries[0];
631 iEnd=PtrAdd(p,iPageSize);
636 void CDbOrderByStage::CKeys::KeyComplete(TAny* aEnd)
638 // Key is complete, prepare for the next one
641 __ASSERT(aEnd==Align4(aEnd));
642 __ASSERT(iNext<=aEnd&&aEnd<=iEnd);
643 iNext=STATIC_CAST(TKey*,aEnd);
646 #ifdef _CHECK_ORDERING
647 void CDbOrderByStage::CKeys::VerifyOrderingL(const HOrdering& aOrder)
649 // this performs a full O(N*N) comparison of the record set
650 if (iKeys.Count()==0)
652 TKey* const* data=&iKeys[0];
653 for (TInt ii=iKeys.Count();--ii>=0;)
655 for (TInt jj=iKeys.Count();--jj>=0;)
657 TInt rr=aOrder.CompareL(data[ii]->iKey,data[jj]->iKey);
659 __ASSERT_ALWAYS(rr==0,User::Invariant());
661 __ASSERT_ALWAYS(rr<=0,User::Invariant());
663 __ASSERT_ALWAYS(rr>=0,User::Invariant());
669 // Class CDbOrderByStage
671 inline const RDbTableRow& CDbOrderByStage::Row()
673 inline CDbOrderByStage::CKeys& CDbOrderByStage::Keys()
674 {__ASSERT(iKeys);return *iKeys;}
675 inline const CDbOrderByStage::HOrdering& CDbOrderByStage::Order()
676 {__ASSERT(iOrder);return *iOrder;}
678 CDbOrderByStage::CDbOrderByStage(const RDbTableRow& aRow)
679 : CDbBasicWindowStage(KDbUnlimitedWindow),iRow(aRow)
681 __ASSERT(iState==EReset);
684 CDbOrderByStage::~CDbOrderByStage()
690 void CDbOrderByStage::ConstructL(const CDbKey& aOrderBy)
692 // Build the key structures to support the partial and complete ordering
695 iOrder=HOrdering::NewL(Row().Table(),aOrderBy);
698 void CDbOrderByStage::Reset()
700 // Reset the window to initial state
706 CDbBasicWindowStage::Reset();
709 TBool CDbOrderByStage::ReadL(TInt& aWork,TDbPosition aNext)
711 // Read some more record keys
714 TDbRecordId id(KDbNullRecordIdValue);
716 while ((r=CDbDataStage::GotoL(aWork,aNext,id))==ESuccess)
718 CDbDataStage::ReadRowL(id);
719 Keys().AddL(id,Row(),Order());
727 __LEAVE(KErrNotReady);
735 TBool CDbOrderByStage::DoEvaluateL(TInt& aWork)
737 // Build the key collection, and finally sort it
747 __LEAVE(KErrNotReady);
750 iKeys=new(ELeave) CKeys(Order().MaxSize());
751 // drop through to EEvaluate
753 if (ReadL(aWork,s==EReset ? EDbFirst : EDbNext))
758 // drop through to ESort
760 Keys().SortL(Order());
761 #ifdef _CHECK_ORDERING
762 Keys().VerifyOrderingL(Order());
764 // drop through to EAdd
766 Keys().GetRecordsL(iRecords);
769 // drop through to EComplete
776 TBool CDbOrderByStage::Unevaluated()
778 // Return whether it is worth Evaluating
781 if (iState==EComplete)
782 return CDbDataStage::Unevaluated();
783 return iState!=EFailed;
787 // Class CDbReorderWindowStage
789 CDbReorderWindowStage::CDbReorderWindowStage()
790 : CDbBasicWindowStage(KDbUnlimitedWindow),iRows(EGranularity)
792 __ASSERT(iState==EReset);
795 CDbReorderWindowStage::~CDbReorderWindowStage()
800 void CDbReorderWindowStage::Reset()
802 // Reset the window to initial state
807 CDbBasicWindowStage::Reset();
810 TBool CDbReorderWindowStage::ReadL(TInt& aWork,TDbPosition aNext)
812 // Read some more row ids
815 TDbRecordId id(KDbNullRecordIdValue);
817 while ((r=CDbDataStage::GotoL(aWork,aNext,id))==ESuccess)
819 __LEAVE_IF_ERROR(iRows.Append(id.Value()));
827 __LEAVE(KErrNotReady);
835 TBool CDbReorderWindowStage::DoEvaluateL(TInt& aWork)
837 // Build the key collection, and finally sort it
847 __LEAVE(KErrNotReady);
850 if (ReadL(aWork,s==EReset ? EDbFirst : EDbNext))
858 iRecords.AppendL((const TDbRecordId*)&iRows[0],iRows.Count());
861 // coverity[fallthrough]
868 TBool CDbReorderWindowStage::Unevaluated()
870 // Return whether it is worth Evaluating
873 if (iState==EComplete)
874 return CDbDataStage::Unevaluated();
875 return iState!=EFailed;