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.
14 // SQL Like predicate node
20 inline TUint8* TextCopy(TUint8* aDest,const TUint8* aSrc,TInt aLen)
21 {return Mem::Copy(aDest,aSrc,aLen);}
22 inline TUint16* TextCopy(TUint16* aDest,const TUint16* aSrc,TInt aLen)
23 {return (TUint16*)Mem::Copy(aDest,aSrc,aLen<<1);}
25 // template Class HMatcherPattern
27 template <class T,class D>
28 inline HMatcherPattern<T,D>* HMatcherPattern<T,D>::Realloc(HMatcherPattern<T,D>* aThis,TInt aSize)
29 {return STATIC_CAST(TThis*,User::ReAlloc(aThis,_FOFF(TThis,iPattern[aSize])));}
32 Creates a HMatcherPattern that converts the pattern into more manageable pieces
33 based on a delimeter that is the wildcard * (asterisk).
35 Characters between * (asterisks) must not number greater than KMaxSegmentLength.
36 If only one * exists then the length is taken from the start and end of pattern.
37 If these segments contain ? (question mark) wildcard than they must not number
38 greater than KMaxSegmentLength-2.
40 The user is responsible for the returned pointer's memory.
42 @param aPattern The search pattern to match.
43 @param aEscape ESCAPE clause
44 @leave KErrNoMemory if no more memory is available.
45 @leave KErrArgument if the search pattern segment length is greater
46 than KMaxSegmentLength,
47 @return The newly constructed pattern pointer.
48 @see KMaxSegmentLength
50 template <class T,class D>
51 HMatcherPattern<T,D>* HMatcherPattern<T,D>::NewL(const D& aPattern, TBool aEscape)
54 TInt r=Construct(self,aPattern,aEscape);
63 template <class T,class D>
64 TInt HMatcherPattern<T,D>::MatchL(const D& aDes,const TTextOps& aTextOp) const
66 // Test the pattern against the supplied descriptor
69 TDesMatcher<T,D> matcher(aDes);
70 return matcher.MatchL(*this,aTextOp);
73 template <class T,class D>
74 TInt HMatcherPattern<T,D>::MatchL(MStreamBuf& aBuf,TInt aLength,const TTextOps& aTextOp) const
76 // Test the pattern against the supplied stream
79 TStreamMatcher<T,D> matcher(aBuf,aLength);
80 return matcher.MatchL(*this,aTextOp);
84 Breaks up a given search pattern into manageable segments.
86 The pattern is broken into segments. These segments are organised
87 from the segment delimeter that is the wildcard * (asterisk).
88 So in effect a segment may contain other wildcards (i.e. ?) or
89 the entire pattern (no embedded asterisks) as these at least MUST match.
91 The cell is created and updated with the segments type and length, and also
94 Not including asterisks, the segment must be no longer than length KMaxSegmentLength.
95 However, if the segment has a ? (question mark) wildcard within it then the
96 segment must be no longer than length KMaxSegmentLength-2.
98 This is due to the way the Find and Match functions of TDes work. Match understands
99 wildcards whilst Find does not.
101 The match algorithm depends on KMaxSegmentLength being smaller than the
102 text read buffer, and for efficiency should not be more than half the size
103 of the read buffer. Also KMaxSegmentLength must currently fit into 8 bits,
104 as it is embedded into a single character in an 8-bit text matching buffer.
105 [So increasing KMaxSegmentLength is not a simple exercise.]
107 The search is repeated for each segment within the pattern. Increasing the cell
108 size and inserting each segment and associated segment data.
110 On reaching the end of the pattern this data is passed back to the user.
111 The user is responsible for this returned pointers memory.
113 @param aCell On return points to a cell matching this pattern.
114 @param aPattern The search pattern to match.
115 @param aEscape ESCAPE clause
116 @return KErrNoMemory if no memory is available to create or
117 increase the size of a cell,
118 KErrArgument if the search pattern segment length is greater
119 than KMaxSegmentLength,
120 KErrNone if the match was successful.
121 @see KMaxSegmentLength
123 template <class T,class D>
124 TInt HMatcherPattern<T,D>::Construct(HMatcherPattern<T,D>*& aCell,const D& aPattern,TBool aEscape)
126 const T* pM=aPattern.Ptr();
127 const T* const eM=pM+aPattern.Length();
128 TInt size=(eM-pM)+KPatternGranularity;
129 TThis* self=Realloc(0,size);
133 T* seg=&self->iPattern[0];
134 const T* end=&self->iPattern[size];
135 TInt term; // terminating code
144 //Following code is for the implementation of limited-ESCAPE-clause
148 TInt len = aPattern.Length();
149 if (len>KMaxSegmentLength)
153 seg=TextCopy(seg,here,len);
165 } while (m==KMatchAny);
173 while (++pM<eM && *pM==KMatchOne) {;}
195 { // include '*'s in segment for matching
203 if (len>KMaxSegmentLength)
210 TInt newsize=end-&self->iPattern[0]+KPatternGranularity;
211 self=Realloc(self,newsize);
214 seg+=&self->iPattern[0]-&aCell->iPattern[0];
215 end=&self->iPattern[newsize];
218 if (type==(EMiddle|EWild))
224 seg=TextCopy(seg,here,len);
228 aCell=Realloc(self,seg-&self->iPattern[0]);
234 template <class T,class D>
235 TBool TMatcher<T,D>::MatchL(const HMatcherPattern<T,D>& aPattern,const TTextOps& aTextOp)
237 const T* pM=&aPattern.iPattern[0];
239 const T* pB=UnderflowL(eB);
243 switch (segment&EMask)
259 pB=UnderflowL(eB)+skip;
268 //Following code is for the implementation of limited-ESCAPE-clause
276 if (aTextOp.Find(pB,bLen,pM,len)<0)
290 if (aTextOp.Match(pB,len,pM,len)<0)
295 if (aTextOp.Compare(pB,len,pM,len)!=0)
308 match-=2; // the number of characters to match
311 pB=UnderflowL(eB,left);
317 TInt pos=segment&EWild ? aTextOp.Match(pB,bLen,pM,len) : aTextOp.Find(pB,bLen,pM,len);
323 // not found, next chunk of data please.
324 pB=UnderflowL(eB,match-1);
329 case EEnd: // match the last segment
336 pB=UnderflowL(eB,left);
341 pB=UnderflowL(eB,len);
344 if (aTextOp.Match(pB,len,pM,len)<0)
349 if (aTextOp.Compare(pB,len,pM,len)!=0)
360 template <class T,class D>
361 inline TDesMatcher<T,D>::TDesMatcher(const D& aDes)
362 {iEnd=(iPtr=aDes.Ptr())+aDes.Length();}
364 template <class T,class D>
365 const T* TDesMatcher<T,D>::UnderflowL(const T*& aEnd,TInt aRetain)
367 const T* ptr=iPtr-aRetain;
372 // Class TStreamMatcher
374 template <class T,class D>
375 inline TStreamMatcher<T,D>::TStreamMatcher(MStreamBuf& aStreamBuf,TInt aLength)
376 :iStream(&aStreamBuf),iRemain(aLength),iEnd(&iBuffer[EBufSize])
379 template <class T,class D>
380 const T* TStreamMatcher<T,D>::UnderflowL(const T*& aEnd,TInt aRetain)
383 { // no more stream data, don't move etc.
391 rp=TextCopy(rp,iEnd-aRetain,aRetain);
392 TInt read=Min(EBufSize-aRetain,iRemain);
393 iStream.ReadL(rp,read);
402 void RSqlLiteral::ToPattern8L(TBool aEscape)
404 // Convert a Buf8 to an 8-bit pattern
407 __ASSERT(iType==EBuf8);
408 HMatcherPattern8* pattern=HMatcherPattern8::NewL(Text8(),aEscape);
409 User::Free(iVal.iAlloc);
414 void RSqlLiteral::ToPattern16L(TBool aEscape)
416 // Convert a Buf8 to an 8-bit pattern
419 __ASSERT(iType==EBuf16);
420 HMatcherPattern16* pattern=HMatcherPattern16::NewL(Text16(),aEscape);
421 User::Free(iVal.iAlloc);
427 // Class CSqlLikePredicate
429 LOCAL_C TInt MatchLength(const TText* aPtr,const TText* aEnd)
444 //Following code is for the implementation of limited-ESCAPE-clause
445 LOCAL_C TInt MatchLengthEscape(const TText* aPtr,const TText* aEnd)
461 CSqlLikePredicate::CSqlLikePredicate(TType aType,const TDesC& aColumn,const RSqlLiteral& aPattern)
462 : CSqlLiteralNode(aType,aColumn,aPattern)
464 __ASSERT(aType==ELike||aType==ENotLike);
467 void CSqlLikePredicate::BindL(const RDbTableRow& aSource)
469 // Compile the Pattern for LongText columns and evaluate the match length
473 if(iIsEscape)//Following code is for the implementation of limited-ESCAPE-clause
475 matchsize = MatchLengthEscape(Value().Ptr(),Value().End());
479 matchsize = MatchLength(Value().Ptr(),Value().End());
481 CSqlLiteralNode::BindL(aSource);
484 default: // type mismatch: should be caught by Literal::Bind
488 case EDbColLongText8:
489 Value().ToPattern8L(iIsEscape);
491 case EDbColLongText16:
492 Value().ToPattern16L(iIsEscape);
499 iMatchSize=matchsize;
502 TBool CSqlLikePredicate::EvaluateL(const TTextOps& aTextOp) const
505 TDbColType type=ColType();
506 TDbColumnC column(Column());
508 if (TDbCol::IsLong(type))
510 const TDbBlob& blob=column.Blob();
511 if (blob.Size()<iMatchSize)
513 else if (blob.IsInline())
514 match=type==EDbColLongText8 ?
515 Value().Pattern8().MatchL(blob.PtrC8(),aTextOp) :
516 Value().Pattern16().MatchL(blob.PtrC16(),aTextOp);
519 MStreamBuf& buf=StreamLC(blob);
520 match=type==EDbColLongText8 ?
521 Value().Pattern8().MatchL(buf,blob.Size(),aTextOp) :
522 Value().Pattern16().MatchL(buf,blob.Size()>>1,aTextOp);
523 CleanupStack::PopAndDestroy();
526 else if (column.Size()<iMatchSize)
530 if(iIsEscape) //Following code is for the implementation of limited-ESCAPE-clause
532 match=type==EDbColText8 ?
533 aTextOp.Find(column.PtrC8(),Value().Text8()) :
534 aTextOp.Find(column.PtrC16(),Value().Text16());
538 match=type==EDbColText8 ?
539 aTextOp.Match(column.PtrC8(),Value().Text8()) :
540 aTextOp.Match(column.PtrC16(),Value().Text16());
546 return NodeType()==ELike ? match : !match;