os/persistentdata/persistentstorage/dbms/utable/UT_ORDER.CPP
author sl@SLION-WIN7.fritz.box
Fri, 15 Jun 2012 03:10:57 +0200
changeset 0 bde4ae8d615e
permissions -rw-r--r--
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".
     7 //
     8 // Initial Contributors:
     9 // Nokia Corporation - initial contribution.
    10 //
    11 // Contributors:
    12 //
    13 // Description:
    14 // Order-by stage for the cursor data "pipeline"
    15 // 
    16 //
    17 
    18 #include "UT_STD.H"
    19 #include "D32COMP.H"
    20 #include <d32dbmsconstants.h>
    21 
    22 #ifdef _DEBUG
    23 #define _CHECK_ORDERING		// forces a full test of the ordering after sorting
    24 #endif
    25 
    26 #define MAX( a, b ) ( (a) > (b) ? (a) : (b) )
    27 
    28 /* The key structure.
    29 
    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)
    33 
    34   Text8 columns			<byte n><n ASCII characters>
    35   n is the [character] length of the column
    36 
    37   Text16 columns		<short n><n UNICODE characters>
    38   n is the [character] length of the column
    39 
    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
    43 
    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
    47 
    48 */
    49 
    50 
    51 class CDbOrderByStage::HOrdering
    52 	{
    53 private:
    54 	struct TKeyCol
    55 		{
    56 		TDbColNo iOrdinal;
    57 		TUint8 iType;
    58 		TUint8 iOrder;
    59 		TUint8 iLength;
    60 		};
    61 	struct TBlobKey
    62 		{
    63 		enum {ETrunc=(TInt)0x80000000};
    64 		enum {ETruncSize=32};		// store 32 bytes of truncated BLOBs
    65 	public:
    66 		inline TInt Size() const;
    67 	public:
    68 		TInt iSize;
    69 		union {TUint8 iData8[ETruncSize]; TUint16 iData16[ETruncSize>>1];};
    70 		TDbBlobId iId;
    71 		};
    72 public:
    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;
    76 	TInt MaxSize() const;
    77 private:
    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;
    82 private:
    83 	CDbTable& iTable;
    84 	const TTextOps& iTextOps;
    85 	const TKeyCol* iEndOfKeys;
    86 	TKeyCol iKeys[1];	// one or more keys
    87 	};
    88 
    89 // Class HDbOrdering
    90 
    91 inline TInt CDbOrderByStage::HOrdering::TBlobKey::Size() const
    92 	{
    93 	TInt size=_FOFF(TBlobKey,iData8[iSize+3]);
    94 	return (size&ETrunc) ? sizeof(TBlobKey) : size&~3;
    95 	}
    96 
    97 inline CDbOrderByStage::HOrdering::HOrdering(TInt aCount,TDbTextComparison aComparison,CDbTable& aTable) :
    98 	iTable(aTable),
    99 	iTextOps(TTextOps::Ops(aComparison)),
   100 	iEndOfKeys(&iKeys[aCount])
   101 	{
   102 	}
   103 
   104 //
   105 // Construct the ordering based on the key definition (must be valid for the table)
   106 // and the column set provided
   107 //
   108 CDbOrderByStage::HOrdering* CDbOrderByStage::HOrdering::NewL(CDbTable& aTable,const CDbKey& aKey)
   109 	{
   110 	TInt count=aKey.Count();
   111 	__ASSERT(count>0);
   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)
   117 		{
   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)
   125 			{
   126 		default:
   127 			break;
   128 		case EDbColText8:
   129 		case EDbColText16:
   130 			{
   131 			TInt l=col.iMaxLength;
   132 			if (l<256)
   133 				{
   134 				pKey->iLength=TUint8(l);
   135 				break;
   136 				}
   137 			}
   138 			// fall through to argument error
   139 		case EDbColBinary:
   140 		case EDbColLongBinary:
   141 			__LEAVE(KErrArgument);
   142 			}
   143 		}
   144 	CleanupStack::Pop();
   145 	return self;
   146 	}
   147 
   148 TInt CDbOrderByStage::HOrdering::MaxSize() const
   149 	{
   150 	TInt maxsize=0;
   151 	const TKeyCol* const end=iEndOfKeys;
   152 	const TKeyCol* key=&iKeys[0];
   153 	__ASSERT(key<end);
   154 	do	{
   155 		TInt s;
   156 		switch (key->iType)
   157 			{
   158 		default:
   159 			s=sizeof(TUint32);
   160 			break;
   161 		case EDbColInt64:
   162 			s=sizeof(TInt64);
   163 			break;
   164 		case EDbColDateTime:
   165 			s=sizeof(TTime);
   166 			break;
   167 		case EDbColReal64:
   168 			s=sizeof(TReal64);
   169 			break;
   170 		case EDbColText8:
   171 			s=Align4(key->iLength+1);
   172 			break;
   173 		case EDbColText16:
   174 			s=Align4((key->iLength<<1)+1);
   175 			break;
   176 		case EDbColLongText8:
   177 		case EDbColLongText16:
   178 			s=MAX(__Align8(sizeof(TUint32)+KDbMaxInlineBlobSize),sizeof(TBlobKey));
   179 			break;
   180 			}
   181 		maxsize+=s;
   182 		} while (++key<end);
   183 	return maxsize;
   184 	}
   185 
   186 MStreamBuf* CDbOrderByStage::HOrdering::BlobLC(TDbBlobId aId,TDbColType aType) const
   187 	{
   188 	return iTable.BlobsL()->ReadLC(aId,aType);
   189 	}
   190 
   191 //
   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
   195 //
   196 TAny* CDbOrderByStage::HOrdering::EntryL(TAny* aPtr,const RDbTableRow& aRow) const
   197 	{
   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;
   202 	__ASSERT(key<end);
   203 	do	{
   204 		const TDbCell* cell=aRow.ColCell(key->iOrdinal);
   205 		TDbColType type=TDbColType(key->iType);
   206 		if (cell->Length()==0)
   207 			{	// null column
   208 			const TUint K64BitColumnMask=(1u<<EDbColInt64)|(1u<<EDbColReal64)|(1u<<EDbColDateTime);
   209 			*ptr++=0;
   210 			if (K64BitColumnMask&(1u<<type))
   211 				*ptr++=0;		// 8-byte column
   212 			continue;
   213 			}
   214 		switch (type)
   215 			{
   216 		default:
   217 			__ASSERT(0);
   218 		case EDbColBit:
   219 		case EDbColUint8:
   220 		case EDbColUint16:
   221 		case EDbColUint32:
   222 		case EDbColInt8:
   223 		case EDbColInt16:
   224 		case EDbColInt32:
   225 		case EDbColReal32:
   226 			*ptr++=*(const TUint32*)cell->Data();
   227 			break;
   228 		case EDbColInt64:
   229 		case EDbColDateTime:
   230 		case EDbColReal64:
   231 			{
   232 			const TUint32* data=(const TUint32*)cell->Data();
   233 			*ptr++=*data++;
   234 			*ptr++=*data;
   235 			}
   236 			break;
   237 		case EDbColText8:
   238 			{
   239 			TInt size=cell->Length();
   240 			*(TUint8*)ptr=TUint8(size);
   241 			ptr=(TUint32*)Align4(Mem::Copy(PtrAdd(ptr,1),cell->Data(),size));
   242 			}
   243 			break;
   244 		case EDbColText16:
   245 			{
   246 			TInt size=cell->Length();
   247 			*(TUint16*)ptr=TUint16(size>>1);
   248 			ptr=(TUint32*)Align4(Mem::Copy(PtrAdd(ptr,2),cell->Data(),size));
   249 			}
   250 			break;
   251 		case EDbColLongText8:
   252 		case EDbColLongText16:
   253 			{
   254 			const TDbBlob& blob=*(const TDbBlob*)cell->Data();
   255 			TInt size=blob.Size();
   256 			if (blob.IsInline())
   257 				{
   258 				*ptr++=size;
   259 				ptr=(TUint32*)Align4(Mem::Copy(ptr,blob.Data(),size));
   260 				}
   261 			else
   262 				{
   263 				TInt rsize=size;
   264 				if (size>TBlobKey::ETruncSize)
   265 					{
   266 					size|=TBlobKey::ETrunc;
   267 					rsize=TBlobKey::ETruncSize;
   268 					}
   269 				*ptr++=size;
   270 				BlobLC(blob.Id(),TDbColType(key->iType))->ReadL(ptr,rsize);
   271 				CleanupStack::PopAndDestroy();
   272 				ptr=Align4(PtrAdd(ptr,rsize));
   273 				if (size&TBlobKey::ETrunc)
   274 					*ptr++=blob.Id();
   275 				}
   276 			}
   277 			break;
   278 			}
   279 		} while (++key<end);
   280 	return ptr;
   281 	}
   282 
   283 TInt CDbOrderByStage::HOrdering::CompareLongText8L( const CDbOrderByStage::HOrdering::TBlobKey& aLeft, const CDbOrderByStage::HOrdering::TBlobKey& aRight ) const
   284 	{
   285 	TUint sizel = aLeft.iSize;
   286 	TUint sizer = aRight.iSize;
   287 	TUint s = sizel;
   288 	if ( sizer < s )
   289 		s = sizer;
   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 );
   293 	if ( rr )
   294 		return rr;
   295 //
   296 // matches up to the same-length inlined data...
   297 // now it gets interesting
   298 //
   299 	if ( ( ( sizel | sizer ) & static_cast<TUint>( TBlobKey::ETrunc ) ) == 0 )
   300 		return sizel - sizer;		// neither are truncated
   301 	if ( s == sizel )
   302 		return -1;		// left completely match against truncated right
   303 	if ( s == sizer )
   304 		return 1;		// right completely matched against truncated left
   305 
   306 	s = Min( TInt( sizel & ~TBlobKey::ETrunc ), TInt( sizer & ~TBlobKey::ETrunc ) );		// minimum real length
   307 	if ( sizel & static_cast<TUint>( TBlobKey::ETrunc ) )
   308 		{
   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();
   314 			}
   315 		else	// left side only out-of-line
   316 			rr = Comp::Compare8L( bufL, aRight.iData8, s, iTextOps );
   317 		}
   318 	else
   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 );
   322 		}
   323 	CleanupStack::PopAndDestroy();
   324 	return rr ? rr : ( sizel & ~TBlobKey::ETrunc ) - ( sizer & ~TBlobKey::ETrunc );
   325 	}
   326 
   327 TInt CDbOrderByStage::HOrdering::CompareLongText16L( const CDbOrderByStage::HOrdering::TBlobKey& aLeft, const CDbOrderByStage::HOrdering::TBlobKey& aRight ) const
   328 	{
   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;
   333 	
   334 	if (!(truncL | truncR)) // neither side is truncated, compare full strings
   335 		{		
   336 		return iTextOps.Order( aLeft.iData16, sizeL>>1, aRight.iData16, sizeR>>1 );
   337 		}
   338 		
   339 	TInt result;
   340 	TUint sizeMin = Min( TInt(sizeL), TInt(sizeR) ); // minimum real length
   341 	if ( truncL )
   342 		{
   343 		MStreamBuf& bufL = *BlobLC( aLeft.iId, EDbColLongText16 );
   344 		if ( truncR )
   345 			{	// both out-of-line
   346 			result = Comp::Compare16L( bufL, *BlobLC( aRight.iId, EDbColLongText16 ), sizeMin, iTextOps );
   347 			CleanupStack::PopAndDestroy();
   348 			}
   349 		else	// left side only out-of-line
   350 			result = Comp::Compare16L( bufL, aRight.iData16, sizeMin, iTextOps );
   351 		}
   352 	else
   353 		{	// right side only out-of-line
   354 		__ASSERT( truncR );
   355 		result = -Comp::Compare16L( *BlobLC( aRight.iId, EDbColLongText16 ), aLeft.iData16, sizeMin, iTextOps );
   356 		}
   357 	CleanupStack::PopAndDestroy();
   358 	return result ? result : ( sizeL ) - ( sizeR );
   359 	}
   360 
   361 TInt CDbOrderByStage::HOrdering::CompareL(const TAny* aLeft,const TAny* aRight) const
   362 //
   363 // compare the keys
   364 //
   365 	{
   366 	
   367     __ASSERT(aLeft != NULL);
   368     __ASSERT(aRight != NULL);
   369 
   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];
   374 	TInt rr;
   375 	for (;;)
   376 		{
   377 		switch (key->iType)
   378 			{
   379 		default:
   380 			__ASSERT(0);
   381 		case EDbColBit:
   382 		case EDbColUint8:
   383 		case EDbColUint16:
   384 		case EDbColUint32:
   385 			rr=Comp::Compare(*(const TUint32*)left,*(const TUint32*)right);
   386 			left+=sizeof(TUint32);
   387 			right+=sizeof(TUint32);
   388 			break;
   389 		case EDbColInt8:
   390 		case EDbColInt16:
   391 		case EDbColInt32:
   392 			rr=Comp::Compare(*(const TInt32*)left,*(const TInt32*)right);
   393 			left+=sizeof(TInt32);
   394 			right+=sizeof(TInt32);
   395 			break;
   396 		case EDbColInt64:
   397 			rr=Comp::Compare(*(const TInt64*)left,*(const TInt64*)right);
   398 			left+=sizeof(TInt64);
   399 			right+=sizeof(TInt64);
   400 			break;
   401 		case EDbColDateTime:
   402 			rr=Comp::Compare(*(const TTime*)left,*(const TTime*)right);
   403 			left+=sizeof(TTime);
   404 			right+=sizeof(TTime);
   405 			break;
   406 		case EDbColReal32:
   407 			rr=Comp::Compare(*(const TReal32*)left,*(const TReal32*)right);
   408 			left+=sizeof(TReal32);
   409 			right+=sizeof(TReal32);
   410 			break;
   411 		case EDbColReal64:
   412 			rr=Comp::Compare(*(const TReal64*)left,*(const TReal64*)right);
   413 			left+=sizeof(TReal64);
   414 			right+=sizeof(TReal64);
   415 			break;
   416 		case EDbColText8:
   417 			{
   418 			TInt sizel=*left++;
   419 			TInt sizer=*right++;
   420 			rr=iTextOps.Compare(left,sizel,right,sizer);
   421 			left=Align4(left+sizel);
   422 			right=Align4(right+sizer);
   423 			}
   424 			break;
   425 		case EDbColText16:
   426 			{
   427 			const TUint16* l16=reinterpret_cast<const TUint16*>(left);
   428 			const TUint16* r16=reinterpret_cast<const TUint16*>(right);
   429 			TInt sizel=*l16++;
   430 			TInt sizer=*r16++;
   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));
   434 			}
   435 			break;
   436 		case EDbColLongText8:
   437 			{
   438 			const TBlobKey* ltext=(const TBlobKey*)left;
   439 			const TBlobKey* rtext=(const TBlobKey*)right;
   440 			rr=CompareLongText8L(*ltext,*rtext);
   441 			left+=ltext->Size();
   442 			right+=rtext->Size();
   443 			}
   444 			break;
   445 		case EDbColLongText16:
   446 			{
   447 			const TBlobKey* ltext=(const TBlobKey*)left;
   448 			const TBlobKey* rtext=(const TBlobKey*)right;
   449 			rr=CompareLongText16L(*ltext,*rtext);
   450 			left+=ltext->Size();
   451 			right+=rtext->Size();
   452 			}
   453 			break;
   454 			}
   455 		if (rr!=0)
   456 			break;
   457 		if (++key==end)
   458 			break;
   459 		}
   460 	return key->iOrder==TDbKeyCol::EAsc ? rr : -rr;
   461 	}
   462 
   463 //
   464 
   465 NONSHARABLE_CLASS(CDbOrderByStage::CKeys) : public CBase
   466 	{
   467 public:		// should be private but VC++ 4.0 whinges
   468 	struct TKey
   469 		{
   470 		TUint32 iId;
   471 		TUint8 iKey[4];		// and then some
   472 		};
   473 private:
   474 	enum {EKeysGranularity=32};
   475 	struct TPage
   476 		{
   477 		TPage* iNext;
   478 		TKey iEntries[1];
   479 		};
   480 	enum {EMinPageSize=0x200,EMinKeyElements=2};
   481 public:
   482 	CKeys(TInt aMaxKeySize);
   483 	~CKeys();
   484 //
   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);
   490 #endif
   491 private:
   492 	void SortL(TKey* aData[],TInt aElem,const HOrdering& aOrder);
   493 	TKey& NewKeyL();
   494 	void KeyComplete(TAny* aEnd);
   495 	void Release();
   496 private:
   497 	RPointerArray<TKey> iKeys;
   498 	TInt iMaxKeySize;
   499 	TInt iPageSize;
   500 	TPage* iPages;
   501 	TKey* iNext;
   502 	TKey* iEnd;
   503 	};
   504 
   505 // Class CDbOrderByStage::CKeys
   506 
   507 CDbOrderByStage::CKeys::CKeys(TInt aMaxKeySize)
   508 	:iKeys(EKeysGranularity),iMaxKeySize(_FOFF(TKey,iKey[aMaxKeySize])),
   509 	 iPageSize(Max(EMinPageSize+iMaxKeySize,iMaxKeySize*EMinKeyElements))
   510 	{}
   511 
   512 CDbOrderByStage::CKeys::~CKeys()
   513 	{
   514 	iKeys.Close();
   515 	Release();
   516 	}
   517 
   518 void CDbOrderByStage::CKeys::SortL(CDbOrderByStage::CKeys::TKey* aData[],TInt aElem,const HOrdering& aOrder)
   519 //
   520 // Sort the array of row keys
   521 // Uses Heap-sort
   522 //
   523 	{
   524 	__ASSERT(aElem>1);
   525 	TInt heap=aElem>>1;
   526 	--aElem;
   527 	for (;;)
   528 		{
   529 		TKey* key;
   530 		if (heap!=0)
   531 			key=aData[--heap];
   532 		else
   533 			{
   534 			key=aData[aElem];
   535 			aData[aElem]=aData[0];
   536 			if (--aElem==0)
   537 				{
   538 				aData[0]=key;
   539 				break;
   540 				}
   541 			}
   542 		TInt ix=heap;
   543 		TInt parent;
   544 		for (parent=ix;;parent=ix)
   545 			{
   546 			ix=(ix<<1)+1;
   547 			if (ix<=aElem)
   548 				{
   549 				if (ix<aElem && aOrder.CompareL(aData[ix]->iKey,aData[ix+1]->iKey)<0)
   550 					++ix;
   551 				}
   552 			else
   553 				break;
   554 			if (aOrder.CompareL(aData[ix]->iKey,key->iKey)<=0)
   555 				break;
   556 			aData[parent]=aData[ix];
   557 			}
   558 		aData[parent]=key;
   559 		}
   560 	}
   561 
   562 void CDbOrderByStage::CKeys::AddL(TDbRecordId aRecordId,const RDbTableRow& aRow,const HOrdering& aOrder)
   563 //
   564 // Create a key for the row and store it
   565 //
   566 	{
   567 	TKey& key=NewKeyL();
   568 	key.iId=aRecordId.Value();
   569 	TAny* end=aOrder.EntryL(&key.iKey[0],aRow);
   570 	__LEAVE_IF_ERROR(iKeys.Append(&key));
   571 	KeyComplete(end);
   572 	}
   573 
   574 void CDbOrderByStage::CKeys::SortL(const HOrdering& aOrder)
   575 	{
   576 	TInt elem=iKeys.Count();
   577 	if (elem>1)
   578 		SortL(&iKeys[0],elem,aOrder);
   579 	}
   580 
   581 void CDbOrderByStage::CKeys::GetRecordsL(CArrayFix<TDbRecordId>& aRecords)
   582 	{
   583 	TInt elem=iKeys.Count();
   584 	if (elem==0)
   585 		return;
   586 	TKey** const base=&iKeys[0];
   587 	TKey** ptr=base+elem;
   588 	do	{
   589 		--ptr;
   590 		*REINTERPRET_CAST(TDbRecordId*,ptr)=(*ptr)->iId;
   591 		} while (ptr>base);
   592 	Release();		// discard key memory before adding records
   593 	aRecords.InsertL(0,REINTERPRET_CAST(const TDbRecordId*,base),elem);
   594 	iKeys.Reset();
   595 	}
   596 
   597 void CDbOrderByStage::CKeys::Release()
   598 //
   599 // Discard all of the allocated pages
   600 //
   601 	{
   602 	TPage* p=iPages;
   603 	while (p)
   604 		{
   605 		TPage* n=p->iNext;
   606 		User::Free(p);
   607 		p=n;
   608 		}
   609 	iPages=0;
   610 	iNext=iEnd=0;
   611 	}
   612 
   613 CDbOrderByStage::CKeys::TKey& CDbOrderByStage::CKeys::NewKeyL()
   614 //
   615 // Allocate a key entry for the raw key data
   616 //
   617 	{
   618 	TKey* p=iNext;
   619 	if (PtrAdd(p,iMaxKeySize)>iEnd)
   620 		{	// not enough space for a maximum key
   621 		TPage* page=iPages;
   622 		if (page)
   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
   626 			}
   627 		page=(TPage*)User::AllocL(_FOFF(TPage,iEntries)+iPageSize);
   628 		page->iNext=iPages;
   629 		iPages=page;
   630 		p=iNext=&page->iEntries[0];
   631 		iEnd=PtrAdd(p,iPageSize);
   632 		}
   633 	return *p;
   634 	}
   635 
   636 void CDbOrderByStage::CKeys::KeyComplete(TAny* aEnd)
   637 //
   638 // Key is complete, prepare for the next one
   639 //
   640 	{
   641 	__ASSERT(aEnd==Align4(aEnd));
   642 	__ASSERT(iNext<=aEnd&&aEnd<=iEnd);
   643 	iNext=STATIC_CAST(TKey*,aEnd);
   644 	}
   645 
   646 #ifdef _CHECK_ORDERING
   647 void CDbOrderByStage::CKeys::VerifyOrderingL(const HOrdering& aOrder)
   648 	{
   649 // this performs a full O(N*N) comparison of the record set
   650 	if (iKeys.Count()==0)
   651 		return;
   652 	TKey* const* data=&iKeys[0];
   653 	for (TInt ii=iKeys.Count();--ii>=0;)
   654 		{
   655 		for (TInt jj=iKeys.Count();--jj>=0;)
   656 			{
   657 			TInt rr=aOrder.CompareL(data[ii]->iKey,data[jj]->iKey);
   658 			if (ii==jj)
   659 				__ASSERT_ALWAYS(rr==0,User::Invariant());
   660 			else if (ii<jj)
   661 				__ASSERT_ALWAYS(rr<=0,User::Invariant());
   662 			else
   663 				__ASSERT_ALWAYS(rr>=0,User::Invariant());
   664 			}
   665 		}
   666 	}
   667 #endif
   668 
   669 // Class CDbOrderByStage
   670 
   671 inline const RDbTableRow& CDbOrderByStage::Row()
   672 	{return iRow;}
   673 inline CDbOrderByStage::CKeys& CDbOrderByStage::Keys()
   674 	{__ASSERT(iKeys);return *iKeys;}
   675 inline const CDbOrderByStage::HOrdering& CDbOrderByStage::Order()
   676 	{__ASSERT(iOrder);return *iOrder;}
   677 
   678 CDbOrderByStage::CDbOrderByStage(const RDbTableRow& aRow)
   679 	: CDbBasicWindowStage(KDbUnlimitedWindow),iRow(aRow)
   680 	{
   681 	__ASSERT(iState==EReset);
   682 	}
   683 
   684 CDbOrderByStage::~CDbOrderByStage()
   685 	{
   686 	delete iOrder;
   687 	delete iKeys;
   688 	}
   689 
   690 void CDbOrderByStage::ConstructL(const CDbKey& aOrderBy)
   691 //
   692 // Build the key structures to support the partial and complete ordering
   693 //
   694 	{
   695 	iOrder=HOrdering::NewL(Row().Table(),aOrderBy);
   696 	}
   697 
   698 void CDbOrderByStage::Reset()
   699 //
   700 // Reset the window to initial state
   701 //
   702 	{
   703 	delete iKeys;
   704 	iKeys=0;
   705 	iState=EReset;
   706 	CDbBasicWindowStage::Reset();
   707 	}
   708 
   709 TBool CDbOrderByStage::ReadL(TInt& aWork,TDbPosition aNext)
   710 //
   711 // Read some more record keys
   712 //
   713 	{
   714 	TDbRecordId id(KDbNullRecordIdValue);
   715 	TGoto r;
   716 	while ((r=CDbDataStage::GotoL(aWork,aNext,id))==ESuccess)
   717 		{
   718 		CDbDataStage::ReadRowL(id);
   719 		Keys().AddL(id,Row(),Order());
   720 		aNext=EDbNext;
   721 		}
   722 	switch (r)
   723 		{
   724 	default:
   725 		__ASSERT(0);
   726 	case ESynchFailure:
   727 		__LEAVE(KErrNotReady);
   728 	case EExhausted:
   729 		return ETrue;
   730 	case ENoRow:
   731 		return EFalse;
   732 		}
   733 	}
   734 
   735 TBool CDbOrderByStage::DoEvaluateL(TInt& aWork)
   736 //
   737 // Build the key collection, and finally sort it
   738 //
   739 	{
   740 	TState s=iState;
   741 	iState=EFailed;
   742 	switch (s)
   743 		{
   744 	default:
   745 		__ASSERT(0);
   746 	case EFailed:
   747 		__LEAVE(KErrNotReady);
   748 	case EReset:
   749 		__ASSERT(!iKeys);
   750 		iKeys=new(ELeave) CKeys(Order().MaxSize());
   751 		// drop through to EEvaluate
   752 	case EEvaluate:
   753 		if (ReadL(aWork,s==EReset ? EDbFirst : EDbNext))
   754 			{
   755 			iState=EEvaluate;
   756 			return ETrue;
   757 			}
   758 		// drop through to ESort
   759 	case ESort:
   760 		Keys().SortL(Order());
   761 #ifdef _CHECK_ORDERING
   762 		Keys().VerifyOrderingL(Order());
   763 #endif
   764 		// drop through to EAdd
   765 	case EAdd:
   766 		Keys().GetRecordsL(iRecords);
   767 		delete iKeys;
   768 		iKeys=0;
   769 		// drop through to EComplete
   770 	case EComplete:
   771 		iState=EComplete;
   772 		return EFalse;
   773 		}
   774 	}
   775 
   776 TBool CDbOrderByStage::Unevaluated()
   777 //
   778 // Return whether it is worth Evaluating
   779 //
   780 	{
   781 	if (iState==EComplete)
   782 		return CDbDataStage::Unevaluated();
   783 	return iState!=EFailed;
   784 	}
   785 
   786 
   787 // Class CDbReorderWindowStage
   788 
   789 CDbReorderWindowStage::CDbReorderWindowStage()
   790 	: CDbBasicWindowStage(KDbUnlimitedWindow),iRows(EGranularity)
   791 	{
   792 	__ASSERT(iState==EReset);
   793 	}
   794 
   795 CDbReorderWindowStage::~CDbReorderWindowStage()
   796 	{
   797 	iRows.Close();
   798 	}
   799 
   800 void CDbReorderWindowStage::Reset()
   801 //
   802 // Reset the window to initial state
   803 //
   804 	{
   805 	iRows.Reset();
   806 	iState=EReset;
   807 	CDbBasicWindowStage::Reset();
   808 	}
   809 
   810 TBool CDbReorderWindowStage::ReadL(TInt& aWork,TDbPosition aNext)
   811 //
   812 // Read some more row ids
   813 //
   814 	{
   815 	TDbRecordId id(KDbNullRecordIdValue);
   816 	TGoto r;
   817 	while ((r=CDbDataStage::GotoL(aWork,aNext,id))==ESuccess)
   818 		{
   819 		__LEAVE_IF_ERROR(iRows.Append(id.Value()));
   820 		aNext=EDbNext;
   821 		}
   822 	switch (r)
   823 		{
   824 	default:
   825 		__ASSERT(0);
   826 	case ESynchFailure:
   827 		__LEAVE(KErrNotReady);
   828 	case EExhausted:
   829 		return ETrue;
   830 	case ENoRow:
   831 		return EFalse;
   832 		}
   833 	}
   834 
   835 TBool CDbReorderWindowStage::DoEvaluateL(TInt& aWork)
   836 //
   837 // Build the key collection, and finally sort it
   838 //
   839 	{
   840 	TState s=iState;
   841 	iState=EFailed;
   842 	switch (s)
   843 		{
   844 	default:
   845 		__ASSERT(0);
   846 	case EFailed:
   847 		__LEAVE(KErrNotReady);
   848 	case EReset:
   849 	case EEvaluate:
   850 		if (ReadL(aWork,s==EReset ? EDbFirst : EDbNext))
   851 			{
   852 			iState=EEvaluate;
   853 			return ETrue;
   854 			}
   855 		if (iRows.Count())
   856 			{
   857 			iRows.Sort();
   858 			iRecords.AppendL((const TDbRecordId*)&iRows[0],iRows.Count());
   859 			iRows.Reset();
   860 			}
   861 		// coverity[fallthrough]
   862 	case EComplete:
   863 		iState=EComplete;
   864 		return EFalse;
   865 		}
   866 	}
   867 
   868 TBool CDbReorderWindowStage::Unevaluated()
   869 //
   870 // Return whether it is worth Evaluating
   871 //
   872 	{
   873 	if (iState==EComplete)
   874 		return CDbDataStage::Unevaluated();
   875 	return iState!=EFailed;
   876 	}
   877