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 #include <d32dbmsconstants.h>
20 // Class RDbAccessPlan::TPlan
22 TInt RDbAccessPlan::TPlan::OrderByPlan(const TPlan& aLeft,const TPlan& aRight)
24 TUint lpos=aLeft.Type();
25 TUint rpos=aRight.Type();
26 __ASSERT(lpos!=0 && rpos!=0 ); // should be no table iterators
30 RDbAccessPlan::TPlan::TType RDbAccessPlan::TPlan::Type() const
32 // flag values: 0=A, 1=C, 2=B, 3=D, 8=E/F, 10=G/H, 16=M/N, 18=S/T, 20=I/J, 21=K/L, 22=O/P, 23=Q/R
33 // This determines order of plans
36 static const TUint KPosition[]={EPlanA,EPlanC,EPlanB,EPlanD,0,0,0,0,EPlanEF,0,EPlanGH,0,0,0,0,0,EPlanMN,
37 0,EPlanST,0,EPlanIJ,EPlanKL,EPlanOP,EPlanQR};
38 __ASSERT_ALWAYS(((iFlags&EMask) < (sizeof(KPosition)/sizeof(KPosition[0]))), User::Invariant());
39 return TType(KPosition[iFlags&EMask]);
42 // Class RDbAccessPlan::TBounds
44 RDbAccessPlan::TBounds::TBounds(const TPlan& aPlan)
45 : iLowerPred(aPlan.iLower),iUpperPred(aPlan.iUpper),iLower(0),iUpper(0),iInclusion(0)
49 if (aPlan.iIndex->Key()[0].iOrder==TDbKeyCol::EDesc)
51 TDbLookupKey* t=iLower;
54 iInclusion=(iInclusion>>1)|(iInclusion<<1);
58 void RDbAccessPlan::TBounds::GetLookupKey(const CSqlCompPredicate& aCompPredicate,TDbLookupKey& aLookup)
60 const RSqlLiteral& value=aCompPredicate.Value();
61 switch (aCompPredicate.ColType())
73 aLookup.Add(value.Int64());
77 aLookup.Add(value.Real64());
80 aLookup.Add(value.Time());
84 aLookup.Add(value.Text8());
87 case EDbColLongText16:
88 aLookup.Add(value.Text16());
93 void RDbAccessPlan::TBounds::SetLowerBounds()
95 if (!iLowerPred) // iLower already set to 0
97 GetLookupKey(*iLowerPred,iLowerKey);
98 switch (iLowerPred->NodeType())
102 case CSqlSearchCondition::ELessEqual:
103 case CSqlSearchCondition::ELess:
106 case CSqlSearchCondition::EEqual:
107 case CSqlSearchCondition::EGreaterEqual:
108 iInclusion|=CDbRecordIndex::EIncludeLower;
109 case CSqlSearchCondition::EGreater:
115 void RDbAccessPlan::TBounds::SetUpperBounds()
119 GetLookupKey(*iUpperPred,iUpperKey);
120 switch (iUpperPred->NodeType())
124 case CSqlSearchCondition::EGreaterEqual:
125 case CSqlSearchCondition::EGreater:
128 case CSqlSearchCondition::EEqual:
129 case CSqlSearchCondition::ELessEqual:
130 iInclusion|=CDbRecordIndex::EIncludeUpper;
131 case CSqlSearchCondition::ELess:
137 // Class RDbAccessPlan::CDbCompPredicateList
139 RDbAccessPlan::CDbCompPredicateList* RDbAccessPlan::CDbCompPredicateList::NewLC(CSqlQuery& aQuery,TDbTextComparison aComparison,const CDbTableDef& aTableDef)
141 CDbCompPredicateList* self=new(ELeave) CDbCompPredicateList(aTableDef,aComparison);
142 CleanupStack::PushL(self);
143 CSqlSearchCondition& sc=aQuery.SearchCondition();
144 self->ConstructL(sc);
148 void RDbAccessPlan::CDbCompPredicateList::ConstructL(CSqlSearchCondition& aSearchCondition)
150 // fill the list with valid comp pred's
153 TUint type=Type(aSearchCondition.NodeType());
156 CSqlCompPredicate* cp=aSearchCondition.CompPredicate();
157 const TDesC& colName=cp->ColumnName();
158 if (IsIndexed(colName))
165 CSqlMultiNode* multiNode=aSearchCondition.MultiNode();
166 for (TInt ii=multiNode->Count();ii--;)
168 CSqlSearchCondition* node=multiNode->SubNode(ii);
176 TUint RDbAccessPlan::CDbCompPredicateList::Type(CSqlSearchCondition::TType aType) const
178 // converts CSqlSearchCondition::TType into flag
183 case CSqlSearchCondition::EAnd:
185 case CSqlSearchCondition::ELess:
187 case CSqlSearchCondition::ELessEqual:
189 case CSqlSearchCondition::EEqual:
191 case CSqlSearchCondition::EGreaterEqual:
192 return EGreaterEqual;
193 case CSqlSearchCondition::EGreater:
200 TBool RDbAccessPlan::CDbCompPredicateList::IsIndexed(const TDesC& aColumnName)
202 // Checks if aColumnName is indexed. If its a text column the comparison method should be the same
205 const CDbTableIndexDef* key=iTableDef.Key(aColumnName);
208 const TDbColumnDef* colDef=NULL;
209 TRAPD(errCode, colDef=iTableDef.Columns().ColumnL(aColumnName));
210 if(errCode != KErrNone)
212 if (colDef->Type()>EDbColDateTime && key->Key().Comparison()!=iComparison)
217 CSqlCompPredicate* RDbAccessPlan::CDbCompPredicateList::CompPredicate(TDbColNo aColNo,TUint aType)
219 // Returns first CompPredicate found in the list with required type & col no, and removes from list
222 for (TInt ii=Count();ii--;)
224 CSqlCompPredicate* cp=At(ii);
225 TUint type=Type(cp->NodeType());
226 TDbColNo colNo=cp->ColNo();
227 if (!type&aType || aColNo!=colNo)
237 void Validate::NameL(const TDesC& aName)
239 if (aName.Length()<=KDbMaxName)
242 if (!lex.Eos() && lex.Get().IsAlpha())
250 } while (c.IsAlphaDigit()||c=='_');
253 __LEAVE(KErrBadName);
256 void Validate::UniqueNameL(TDesC const** aNames,TInt aCount,const TDesC& aName)
258 // Ensure that aName is not a duplicate of any of aNames, and add
259 // the new name to the collection. Binary search is used for speed
266 TInt mid=(left+right)>>1;
267 TInt c=aNames[mid]->CompareF(aName);
273 __LEAVE(KErrArgument);
275 Mem::Move(aNames+left+1,aNames+left,(aCount-left)*sizeof(TDesC const*));
279 void Validate::ColSetL(const CDbColSet& aColSet)
281 TDbColSetIter iter(aColSet);
283 __LEAVE(KErrArgument);
284 TDesC const** names=(TDesC const**)User::AllocLC(aColSet.Count()*sizeof(TDesC const*));
287 const TDbCol& col=*iter;
289 UniqueNameL(names,iter.Col()-1,col.iName);
290 TInt type=TInt(col.iType);
291 if (type<TInt(EDbColBit)||type>TInt(EDbColLongBinary))
292 __LEAVE(KErrNotSupported);
293 else if (col.iMaxLength<1&&col.iMaxLength!=KDbUndefinedLength)
294 __LEAVE(KErrArgument);
295 else if (col.iAttributes&~(TDbCol::ENotNull|TDbCol::EAutoIncrement))
296 __LEAVE(KErrNotSupported); // unknown attributes
297 else if (type>EDbColUint32 && col.iAttributes&TDbCol::EAutoIncrement)
298 __LEAVE(KErrArgument); // auto increment on non-integral type
300 CleanupStack::PopAndDestroy();
303 void Validate::KeyL(const CDbKey& aKey,const HDbColumnSet& aColumns)
305 // Check that the key is valid for the table
308 TInt max=aKey.Count()-1;
310 __LEAVE(KErrArgument);
311 for (TInt ii=0;ii<=max;++ii)
313 const TDbKeyCol& kCol=aKey[ii];
314 HDbColumnSet::TIteratorC col=aColumns.ColumnL(kCol.iName);
316 __LEAVE(KErrNotFound);
317 TInt len=kCol.iLength;
318 if (len!=KDbUndefinedLength)
320 if (col->iType<=EDbColDateTime)
321 __LEAVE(KErrArgument);
322 TInt cLen=col->iMaxLength;
326 __LEAVE(KErrNotSupported);
330 __LEAVE(KErrArgument);
331 else if (cLen!=KDbUndefinedLength && cLen<len)
332 __LEAVE(KErrArgument);
337 // Class RDbAccessPlan
339 TUint RDbAccessPlan::FindMatchL(const CDbTableIndexDef* aIndex)
341 // Checks if index best matches the order specified
344 CDbKey& order=iQuery->SortSpecification();
345 TInt count=order.Count();
346 const TDbColumnDef* columnDef = iTable->Def().Columns().ColumnL(order[count-1].iName);
347 __ASSERT(columnDef != NULL);
348 TInt columnLength = columnDef->iMaxLength;
349 const CDbKey& key=aIndex->Key();
351 if (TextKeyL(order) && order.Comparison()!=key.Comparison())
353 TInt kCount=key.Count();
354 for (TInt ii=0,rev=0;;)
356 const TDbKeyCol& kcol=key[ii];
357 const TDbKeyCol& ocol=order[ii];
358 if (kcol.iName.CompareF(ocol.iName)!=0)
360 TInt revcol=kcol.iOrder^ocol.iOrder;
363 else if (rev!=revcol)
365 if (++ii==count) // end of order key
368 if (kcol.iLength!=columnLength)
374 if (ii==kCount) // end of index key
378 { // will provide the right order by, use it
389 TBool RDbAccessPlan::TextKeyL(const CDbKey& anOrder)
391 // return whether any of the keys are text columns
394 const HDbColumnSet& cols=iTable->Def().Columns();
395 TInt count=anOrder.Count();
396 for (TInt ii=0;ii<count;++ii)
398 const TDbColumnDef* columnDef = cols.ColumnL(anOrder[ii].iName);
399 __ASSERT(columnDef != NULL);
401 switch (columnDef->Type())
405 case EDbColLongText8:
406 case EDbColLongText16:
415 CDbTableSource* RDbAccessPlan::TableLC(CDbTableDatabase& aDatabase,const TDesC& aTable)
418 CDbTableSource* source=aDatabase.TableSourceL(aTable);
420 iTable=&source->Table();
421 CleanupStack::PushL(TCleanupItem(Cleanup,this));
425 void RDbAccessPlan::Insert(CDbDataStage* aStage)
428 aStage->SetSource(iSource);
432 void RDbAccessPlan::Cleanup(TAny* aPtr)
434 RDbAccessPlan& self=*STATIC_CAST(RDbAccessPlan*,aPtr);
439 CDbRecordIter* RDbAccessPlan::IteratorL(const TPlan& aPlan)
441 // Returns the right iterator
444 if (aPlan.iFlags&TPlan::EIndex)
445 return iTable->IteratorL(*aPlan.iIndex);
446 if (aPlan.iFlags&TPlan::EBounded)
447 return BoundedIteratorL(aPlan);
448 return iTable->IteratorL();
451 CDbRecordIter* RDbAccessPlan::BoundedIteratorL(const TPlan& aPlan)
453 TBounds bounds(aPlan);
454 CDbRecordIter* iter=iTable->IteratorL(*aPlan.iIndex,bounds.iInclusion,bounds.iLower,bounds.iUpper);
456 iQuery->RemovePredicate(aPlan.iLower);
457 if (aPlan.iUpper && aPlan.iLower!=aPlan.iUpper)
458 iQuery->RemovePredicate(aPlan.iUpper);
462 void RDbAccessPlan::RestrictionL()
464 CDbRestrictStage* restriction=new(ELeave) CDbRestrictStage(iComparison);
466 CSqlSearchCondition* searchcondition=iQuery->AdoptSearchCondition();
467 restriction->SetRestriction(searchcondition);
470 void RDbAccessPlan::OrderByL(const RDbTableRow& aRowBuf)
472 CDbOrderByStage* ordering=new(ELeave) CDbOrderByStage(aRowBuf);
474 ordering->ConstructL(iQuery->SortSpecification());
477 void RDbAccessPlan::ProjectionL()
479 CDbProjectStage* projection=new(ELeave) CDbProjectStage;
481 projection->ConstructL(iQuery->ColumnList(),iTable->Def().Columns());
484 void RDbAccessPlan::WindowL(const TPlan& aPlan,const TDbWindow& aWindow)
486 if (aPlan.iFlags&TPlan::EWindow)
487 Insert(new(ELeave) CDbWindowStage(KDbUnlimitedWindow));
488 else if (aWindow.Size()!=aWindow.ENone && aWindow.Size()!=aWindow.EUnlimited)
489 Insert(new(ELeave) CDbWindowStage(aWindow));
492 TBool RDbAccessPlan::IsIndexIteratorL(TPlan& aPlan,const CDbTableIndexDef* aIndex)
494 // If an index iterator can be used, sets aPlan, else returns EFalse
497 if (!iQuery->HasSortSpecification())
499 TUint ret=FindMatchL(aIndex);
502 // can use index iterator
503 aPlan.iFlags&=~TPlan::EOrder;
505 aPlan.iFlags|=TPlan::EReverse;
506 aPlan.iFlags|=TPlan::EIndex;
511 TBool RDbAccessPlan::IsBoundedIteratorL(TPlan& aPlan,const CDbTableIndexDef* aIndex)
513 // If a bounded iterator can be used, sets aPlan, else returns EFalse
516 if (!iQuery->HasSearchCondition())
518 TDbColNo keyColNo=iTable->Def().Columns().ColNoL(aIndex->Key()[0].iName);
519 CDbCompPredicateList* list=CDbCompPredicateList::NewLC(*iQuery,iComparison,iTable->Def());
520 CSqlCompPredicate* cp=list->CompPredicate(keyColNo,CDbCompPredicateList::EEqual);
521 if (cp==0 && (cp=list->CompPredicate(keyColNo))==0)
523 CleanupStack::PopAndDestroy(); // list
527 TUint type=list->Type(cp->NodeType());
528 aPlan.iLower=type&(CDbCompPredicateList::ELess|CDbCompPredicateList::ELessEqual)?0:cp;
529 aPlan.iUpper=type&(CDbCompPredicateList::EGreater|CDbCompPredicateList::EGreaterEqual)?0:cp;
530 // find other boundary, if present
531 if (list->Count()!=0 && cp->NodeType()!=CSqlSearchCondition::EEqual)
533 TUint nextType=type&(CDbCompPredicateList::ELess|CDbCompPredicateList::ELessEqual) ?
534 CDbCompPredicateList::EGreater|CDbCompPredicateList::EGreaterEqual :
535 CDbCompPredicateList::ELess|CDbCompPredicateList::ELessEqual;
536 CSqlCompPredicate* cp2=list->CompPredicate(keyColNo,nextType);
539 if (nextType&(CDbCompPredicateList::ELess|CDbCompPredicateList::ELessEqual))
545 // check which order the index is in and reverse if descending
546 const TDbKeyCol& key=aIndex->Key()[0];
547 if ((key.iOrder==TDbKeyCol::EDesc && !aPlan.iUpper) || (key.iOrder==TDbKeyCol::EAsc && !aPlan.iLower))
548 aPlan.iFlags|=TPlan::EReverse; // optimise the bounding for forwards iteration
549 if (!list->IsRestriction() && list->Count()==0)
550 aPlan.iFlags&=~TPlan::ERestrict;
551 CleanupStack::PopAndDestroy(); // list
552 aPlan.iFlags|=TPlan::EBounded;
557 void RDbAccessPlan::EvaluatePlansL()
559 // try to find a bounded or index iterator
564 if (iQuery->HasSearchCondition())
565 plan.iFlags|=TPlan::ERestrict;
566 if (iQuery->HasSortSpecification())
567 plan.iFlags|=TPlan::EOrder;
568 // find the right type of iterator
569 TSglQueIterC<CDbTableIndexDef> indexes(iTable->Def().Indexes().AsQue());
570 for (const CDbTableIndexDef* index;(index=indexes++)!=0;)
571 { // only on first column
573 TBool foundIter=IsBoundedIteratorL(newPlan,index);
575 foundIter=IsIndexIteratorL(newPlan,index);
577 EvaluateReorderStage(newPlan,index);
580 EvaluateWindowStage(newPlan);
581 __LEAVE_IF_ERROR(iPlans.Append(newPlan));
586 void RDbAccessPlan::ChoosePlanL(TPlan& aPlan)
589 if (iPlans.Count()==0)
591 CreateTableIteratorPlan(aPlan);
594 TPlan::TType type=iPlans[0].Type();
595 if (!iQuery->HasSearchCondition())
597 __ASSERT(type==TPlan::EPlanEF);
598 GetSmallestKeySize(aPlan,TPlan::EPlanEF);
600 else if (!iQuery->HasSortSpecification())
602 if (type==TPlan::EPlanIJ)
604 GetSmallestKeySize(aPlan,TPlan::EPlanIJ);
605 TInt r=IndexSpanL(aPlan);
606 if (r!=CDbTable::EUnavailableSpan && !iWindow && r>60)
607 CreateTableIteratorPlan(aPlan);
609 else if (type==TPlan::EPlanOP)
611 TInt r=GetTightestRestrictionL(aPlan,TPlan::EPlanOP);
612 if (r==CDbTable::EUnavailableSpan) // no index stats available
613 aPlan=iPlans[0]; // use first O/P as a guess
614 else if ((!iWindow && r>55) || (iWindow && r>60))
615 CreateTableIteratorPlan(aPlan);
620 else if (type==TPlan::EPlanMN)
622 GetSmallestKeySize(aPlan,TPlan::EPlanMN);
623 if (iAccess==RDbRowSet::EUpdatable || iWindow)
624 aPlan.iFlags|=TPlan::EWindow;
626 else if (type==TPlan::EPlanKL)
628 GetSmallestKeySize(aPlan,TPlan::EPlanKL);
629 TInt r=IndexSpanL(aPlan);
630 if (r!=CDbTable::EUnavailableSpan && r>60)
631 { // don't use K plan
632 if (r<75 || GetSmallestKeySize(aPlan,TPlan::EPlanGH)==KErrNotFound)
633 CreateTableIteratorPlan(aPlan);
636 else if (type==TPlan::EPlanQR)
638 TInt r=GetTightestRestrictionL(aPlan,TPlan::EPlanQR);
639 if (r==CDbTable::EUnavailableSpan) // no index stats available
640 aPlan=iPlans[0]; // use first Q/R as a guess
642 CreateTableIteratorPlan(aPlan);
646 __ASSERT(type==TPlan::EPlanGH);
647 // don't use this plan without further data, resort to default
648 CreateTableIteratorPlan(aPlan);
652 void RDbAccessPlan::EvaluateWindowStage(TPlan& aPlan)
656 TUint f=aPlan.iFlags|TPlan::EWindow;
657 if (f&TPlan::EReorder)
659 if (f&TPlan::ERestrict)
661 if (f&TPlan::EOrder) // order-by stage includes window
666 TInt RDbAccessPlan::GetTightestRestrictionL(TPlan& aPlan,TPlan::TType aType)
668 // aPlan is set to the smallest restricted plan with aType
673 for (TInt ii=iPlans.Count();ii--;)
675 if (iPlans[ii].Type()!=aType)
677 TInt t=IndexSpanL(iPlans[ii]);
678 if (t==CDbTable::EUnavailableSpan)
680 __ASSERT(t == span ? (iPlans[ii].iIndex != NULL && plan.iIndex != NULL) : ETrue);
681 //coverity[uninit_use_in_call]
682 if (t<span || (t==span && iPlans[ii].iIndex->Key().Count()<plan.iIndex->Key().Count()))
689 return span!=KMaxTInt?span:CDbTable::EUnavailableSpan;
692 TInt RDbAccessPlan::GetSmallestKeySize(TPlan& aPlan,TPlan::TType aType)
694 // aPlan is set to the plan with smallest index with aType
699 for (TInt ii=iPlans.Count();ii--;)
701 if (iPlans[ii].Type()!=aType)
703 __ASSERT(iPlans[ii].iIndex);
704 TInt count=iPlans[ii].iIndex->Key().Count();
712 return cols!=KMaxTInt?cols:KErrNotFound;
715 TInt RDbAccessPlan::IndexSpanL(const TPlan& aPlan)
717 __ASSERT(aPlan.iIndex);
718 TBounds bounds(aPlan);
719 return iTable->IndexSpanL(*aPlan.iIndex,bounds.iInclusion,bounds.iLower,bounds.iUpper);
722 void RDbAccessPlan::ReducePlans()
724 for (TInt ii=iPlans.Count();--ii>=0;)
726 switch (iPlans[ii].Type())
732 iPlans.Remove(ii); // remove Gw/Hw
735 iPlans[ii].iFlags|=(TPlan::EReorder|TPlan::EOrder); // convert S/T to Q/R
736 EvaluateWindowStage(iPlans[ii]);
740 // explicit conversion required as GCC can't find the TLinearOrder instantiation
741 iPlans.Sort(TLinearOrder<TPlan>(TPlan::OrderByPlan));
744 void RDbAccessPlan::CreateTableIteratorPlan(TPlan& aPlan)
746 aPlan.iFlags&=~(TPlan::EIndex|TPlan::EBounded);
747 if (iQuery->HasSearchCondition())
748 aPlan.iFlags=TPlan::ERestrict;
749 if (iQuery->HasSortSpecification())
750 aPlan.iFlags|=TPlan::EOrder;
751 EvaluateWindowStage(aPlan);
754 void RDbAccessPlan::EvaluateReorderStage(TPlan& aPlan,const CDbTableIndexDef* aIndex)
756 // for a bounded iter with an order by - can the index be used?
759 aPlan.iFlags|=TPlan::EReorder;
760 if (!iQuery->HasSortSpecification())
763 TRAPD(errCode, ret=FindMatchL(aIndex));
764 if (errCode==KErrNone && ret&EMatch)
765 {// do we really need a reorder stage?
766 aPlan.iFlags&=~TPlan::EOrder; // don't need to order it
767 aPlan.iFlags&=~TPlan::EReorder; // don't need a reorder stage
769 aPlan.iFlags|=TPlan::EReverse;
771 aPlan.iFlags&=~TPlan::EReverse;
775 void RDbAccessPlan::PrepareQueryL(CDbTableSource* aSource)
777 if (iQuery->HasSearchCondition())
779 CSqlSearchCondition& sc=iQuery->SearchCondition();
780 sc.BindL(aSource->Row());
782 if (iQuery->HasSortSpecification())
784 CDbKey& order=iQuery->SortSpecification();
785 order.SetComparison(iComparison);
786 Validate::KeyL(order,iTable->Def().Columns());
791 void RDbAccessPlan::BuildLC(CDbTableDatabase& aDatabase,RDbRowSet::TAccess aAccess,const TDbWindow& aWindow)
793 // Prepare the data pipeline
797 CDbTableSource* source=TableLC(aDatabase,iQuery->Table());
799 if (aAccess!=RDbRowSet::EInsertOnly)
800 { // insert only views do not use Iterators, Restrictions or Windows
801 iWindow=aWindow.Size()==aWindow.EUnlimited;
803 PrepareQueryL(source);
807 source->SetIterator(IteratorL(plan));
808 if (plan.iFlags&TPlan::EReorder)
809 Insert(new(ELeave) CDbReorderWindowStage);
810 if (plan.iFlags&TPlan::EReverse)
811 source->ReverseIteratorL();
812 if (plan.iFlags&TPlan::ERestrict)
814 if (plan.iFlags&TPlan::EOrder)
815 OrderByL(source->Row());
816 WindowL(plan,aWindow);
819 if (iQuery->HasColumnList())
823 void RDbAccessPlan::BuildLC(CDbTableDatabase& aDatabase,const TDesC& aTable,RDbRowSet::TAccess aAccess)
825 CDbTableSource* source=TableLC(aDatabase,aTable);
826 if (aAccess!=RDbRowSet::EInsertOnly)
827 source->SetIterator(iTable->IteratorL());