os/persistentdata/persistentstorage/dbms/usql/UQ_LIKE.CPP
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
Update contrib.
sl@0
     1
// Copyright (c) 1998-2009 Nokia Corporation and/or its subsidiary(-ies).
sl@0
     2
// All rights reserved.
sl@0
     3
// This component and the accompanying materials are made available
sl@0
     4
// under the terms of "Eclipse Public License v1.0"
sl@0
     5
// which accompanies this distribution, and is available
sl@0
     6
// at the URL "http://www.eclipse.org/legal/epl-v10.html".
sl@0
     7
//
sl@0
     8
// Initial Contributors:
sl@0
     9
// Nokia Corporation - initial contribution.
sl@0
    10
//
sl@0
    11
// Contributors:
sl@0
    12
//
sl@0
    13
// Description:
sl@0
    14
// SQL Like predicate node
sl@0
    15
// 
sl@0
    16
//
sl@0
    17
sl@0
    18
#include "UQ_STD.H"
sl@0
    19
sl@0
    20
inline TUint8* TextCopy(TUint8* aDest,const TUint8* aSrc,TInt aLen)
sl@0
    21
	{return Mem::Copy(aDest,aSrc,aLen);}
sl@0
    22
inline TUint16* TextCopy(TUint16* aDest,const TUint16* aSrc,TInt aLen)
sl@0
    23
	{return (TUint16*)Mem::Copy(aDest,aSrc,aLen<<1);}
sl@0
    24
sl@0
    25
// template Class HMatcherPattern
sl@0
    26
sl@0
    27
template <class T,class D>
sl@0
    28
inline HMatcherPattern<T,D>* HMatcherPattern<T,D>::Realloc(HMatcherPattern<T,D>* aThis,TInt aSize)
sl@0
    29
	{return STATIC_CAST(TThis*,User::ReAlloc(aThis,_FOFF(TThis,iPattern[aSize])));}
sl@0
    30
sl@0
    31
/**
sl@0
    32
Creates a HMatcherPattern that converts the pattern into more manageable pieces
sl@0
    33
based on a delimeter that is the wildcard * (asterisk).
sl@0
    34
sl@0
    35
Characters between * (asterisks) must not number greater than KMaxSegmentLength.
sl@0
    36
If only one * exists then the length is taken from the start and end of pattern.
sl@0
    37
If these segments contain ? (question mark) wildcard than they must not number
sl@0
    38
greater than KMaxSegmentLength-2.
sl@0
    39
sl@0
    40
The user is responsible for the returned pointer's memory.
sl@0
    41
sl@0
    42
@param			aPattern The search pattern to match.
sl@0
    43
@param			aEscape ESCAPE clause 
sl@0
    44
@leave			KErrNoMemory if no more memory is available.
sl@0
    45
@leave			KErrArgument if the search pattern segment length is greater
sl@0
    46
				than KMaxSegmentLength,
sl@0
    47
@return			The newly constructed pattern pointer.
sl@0
    48
@see			KMaxSegmentLength
sl@0
    49
*/
sl@0
    50
template <class T,class D>
sl@0
    51
HMatcherPattern<T,D>* HMatcherPattern<T,D>::NewL(const D& aPattern, TBool aEscape)
sl@0
    52
	{
sl@0
    53
	TThis* self=0;
sl@0
    54
	TInt r=Construct(self,aPattern,aEscape);
sl@0
    55
	if (r!=KErrNone)
sl@0
    56
		{
sl@0
    57
		User::Free(self);
sl@0
    58
		__LEAVE(r);
sl@0
    59
		}
sl@0
    60
	return self;
sl@0
    61
	}
sl@0
    62
sl@0
    63
template <class T,class D>
sl@0
    64
TInt HMatcherPattern<T,D>::MatchL(const D& aDes,const TTextOps& aTextOp) const
sl@0
    65
//
sl@0
    66
// Test the pattern against the supplied descriptor
sl@0
    67
//
sl@0
    68
	{
sl@0
    69
	TDesMatcher<T,D> matcher(aDes);
sl@0
    70
	return matcher.MatchL(*this,aTextOp);
sl@0
    71
	}
sl@0
    72
sl@0
    73
template <class T,class D>
sl@0
    74
TInt HMatcherPattern<T,D>::MatchL(MStreamBuf& aBuf,TInt aLength,const TTextOps& aTextOp) const
sl@0
    75
//
sl@0
    76
// Test the pattern against the supplied stream
sl@0
    77
//
sl@0
    78
	{
sl@0
    79
	TStreamMatcher<T,D> matcher(aBuf,aLength);
sl@0
    80
	return matcher.MatchL(*this,aTextOp);
sl@0
    81
	}
sl@0
    82
sl@0
    83
/**
sl@0
    84
Breaks up a given search pattern into manageable segments.
sl@0
    85
sl@0
    86
The pattern is broken into segments. These segments are organised
sl@0
    87
from the segment delimeter that is the wildcard * (asterisk).
sl@0
    88
So in effect a segment may contain other wildcards (i.e. ?) or
sl@0
    89
the entire pattern (no embedded asterisks) as these at least MUST match.
sl@0
    90
sl@0
    91
The cell is created and updated with the segments type and length, and also
sl@0
    92
the segment itself.
sl@0
    93
sl@0
    94
Not including asterisks, the segment must be no longer than length KMaxSegmentLength.
sl@0
    95
However, if the segment has a ? (question mark) wildcard within it then the 
sl@0
    96
segment must be no longer than length KMaxSegmentLength-2.
sl@0
    97
sl@0
    98
This is due to the way the Find and Match functions of TDes work. Match understands
sl@0
    99
wildcards whilst Find does not.
sl@0
   100
sl@0
   101
The match algorithm depends on KMaxSegmentLength being smaller than the
sl@0
   102
text read buffer, and for efficiency should not be more than half the size
sl@0
   103
of the read buffer. Also KMaxSegmentLength must currently fit into 8 bits,
sl@0
   104
as it is embedded into a single character in an 8-bit text matching buffer. 
sl@0
   105
[So increasing KMaxSegmentLength is not a simple exercise.]
sl@0
   106
sl@0
   107
The search is repeated for each segment within the pattern. Increasing the cell
sl@0
   108
size and inserting each segment and associated segment data.
sl@0
   109
sl@0
   110
On reaching the end of the pattern this data is passed back to the user.
sl@0
   111
The user is responsible for this returned pointers memory.
sl@0
   112
sl@0
   113
@param			aCell On return points to a cell matching this pattern.
sl@0
   114
@param			aPattern The search pattern to match.
sl@0
   115
@param			aEscape ESCAPE clause 
sl@0
   116
@return			KErrNoMemory if no memory is available to create or
sl@0
   117
				increase the size of a cell,
sl@0
   118
				KErrArgument if the search pattern segment length is greater
sl@0
   119
				than KMaxSegmentLength,
sl@0
   120
				KErrNone if the match was successful.
sl@0
   121
@see			KMaxSegmentLength
sl@0
   122
*/
sl@0
   123
template <class T,class D>
sl@0
   124
TInt HMatcherPattern<T,D>::Construct(HMatcherPattern<T,D>*& aCell,const D& aPattern,TBool aEscape)
sl@0
   125
	{
sl@0
   126
	const T* pM=aPattern.Ptr();
sl@0
   127
	const T* const eM=pM+aPattern.Length();
sl@0
   128
	TInt size=(eM-pM)+KPatternGranularity;
sl@0
   129
	TThis* self=Realloc(0,size);
sl@0
   130
	if (!self)
sl@0
   131
		return KErrNoMemory;
sl@0
   132
	aCell=self;
sl@0
   133
	T* seg=&self->iPattern[0];
sl@0
   134
	const T* end=&self->iPattern[size];
sl@0
   135
	TInt term;	// terminating code
sl@0
   136
	
sl@0
   137
	if (eM==pM)
sl@0
   138
		term=ENull;
sl@0
   139
	else
sl@0
   140
		{
sl@0
   141
		term=EStop;
sl@0
   142
		do
sl@0
   143
			{
sl@0
   144
			//Following code is for the implementation of limited-ESCAPE-clause 
sl@0
   145
			if(aEscape)
sl@0
   146
				{
sl@0
   147
				const T* here=pM;
sl@0
   148
				TInt len = aPattern.Length();
sl@0
   149
				if (len>KMaxSegmentLength)
sl@0
   150
					return KErrArgument;
sl@0
   151
				*seg++=T(EEscape );
sl@0
   152
				*seg++=T(len);
sl@0
   153
				seg=TextCopy(seg,here,len);
sl@0
   154
				break;
sl@0
   155
				}
sl@0
   156
			T m=*pM;
sl@0
   157
			if (m==KMatchAny)
sl@0
   158
				{
sl@0
   159
				term=ESuccess;
sl@0
   160
				do
sl@0
   161
					{
sl@0
   162
					if (++pM==eM)
sl@0
   163
						break;
sl@0
   164
					m=*pM;
sl@0
   165
					} while (m==KMatchAny);
sl@0
   166
				if (pM==eM)
sl@0
   167
					break;
sl@0
   168
				}
sl@0
   169
			const T* here=pM;
sl@0
   170
			TInt type;
sl@0
   171
			if (m==KMatchOne)
sl@0
   172
				{
sl@0
   173
				while (++pM<eM && *pM==KMatchOne) {;}
sl@0
   174
				type=ESkip;
sl@0
   175
				}
sl@0
   176
			else
sl@0
   177
				{
sl@0
   178
				type=0;
sl@0
   179
				while (++pM<eM)
sl@0
   180
					{
sl@0
   181
					m=*pM;
sl@0
   182
					if (m==KMatchAny)
sl@0
   183
						break;
sl@0
   184
					if (m==KMatchOne)
sl@0
   185
						type|=EWild;
sl@0
   186
					}
sl@0
   187
				if (term==EStop)
sl@0
   188
					type|=EBeginning;
sl@0
   189
				else if (pM==eM)
sl@0
   190
					type|=EEnd;
sl@0
   191
				else
sl@0
   192
					{
sl@0
   193
					type|=EMiddle;
sl@0
   194
					if (type&EWild)
sl@0
   195
						{		// include '*'s in segment for matching
sl@0
   196
						--here;
sl@0
   197
						++pM;
sl@0
   198
						}
sl@0
   199
					}
sl@0
   200
				}
sl@0
   201
			*seg++=T(type);
sl@0
   202
			TInt len=pM-here;
sl@0
   203
			if (len>KMaxSegmentLength)
sl@0
   204
				return KErrArgument;
sl@0
   205
			*seg++=T(len);
sl@0
   206
			if (type==ESkip)
sl@0
   207
				len=0;
sl@0
   208
			if (seg+4+len>end)
sl@0
   209
				{
sl@0
   210
				TInt newsize=end-&self->iPattern[0]+KPatternGranularity;
sl@0
   211
				self=Realloc(self,newsize);
sl@0
   212
				if (!self)
sl@0
   213
					return KErrNoMemory;
sl@0
   214
				seg+=&self->iPattern[0]-&aCell->iPattern[0];
sl@0
   215
				end=&self->iPattern[newsize];
sl@0
   216
				aCell=self;
sl@0
   217
				}
sl@0
   218
			if (type==(EMiddle|EWild))
sl@0
   219
				{
sl@0
   220
				*seg++=T(KMatchAny);
sl@0
   221
				++here;
sl@0
   222
				--len;
sl@0
   223
				}
sl@0
   224
			seg=TextCopy(seg,here,len);
sl@0
   225
			} while (pM<eM);
sl@0
   226
		}
sl@0
   227
	*seg++=T(term);
sl@0
   228
	aCell=Realloc(self,seg-&self->iPattern[0]);
sl@0
   229
	return KErrNone;
sl@0
   230
	}
sl@0
   231
	
sl@0
   232
// class TMatcher
sl@0
   233
sl@0
   234
template <class T,class D>
sl@0
   235
TBool TMatcher<T,D>::MatchL(const HMatcherPattern<T,D>& aPattern,const TTextOps& aTextOp)
sl@0
   236
	{
sl@0
   237
	const T* pM=&aPattern.iPattern[0];
sl@0
   238
	const T* eB;
sl@0
   239
	const T* pB=UnderflowL(eB);
sl@0
   240
	for (;;)
sl@0
   241
		{
sl@0
   242
		TInt segment=*pM;
sl@0
   243
		switch (segment&EMask)
sl@0
   244
			{
sl@0
   245
		case ENull:
sl@0
   246
			return pB==eB;
sl@0
   247
		case ESuccess:
sl@0
   248
			return ETrue;
sl@0
   249
		case EStop:
sl@0
   250
			if (pB==eB)
sl@0
   251
				pB=UnderflowL(eB);
sl@0
   252
			return pB==eB;
sl@0
   253
		case ESkip:
sl@0
   254
			{
sl@0
   255
			TInt len=*++pM;
sl@0
   256
			TInt skip=len+pB-eB;
sl@0
   257
			if (skip>0)
sl@0
   258
				{
sl@0
   259
				pB=UnderflowL(eB)+skip;
sl@0
   260
				if (pB>eB)
sl@0
   261
					return EFalse;
sl@0
   262
				}
sl@0
   263
			else
sl@0
   264
				pB+=len;
sl@0
   265
			++pM;
sl@0
   266
			}
sl@0
   267
			break;
sl@0
   268
		//Following code is for the implementation of limited-ESCAPE-clause 
sl@0
   269
		case EEscape:
sl@0
   270
			{
sl@0
   271
			TInt len=*++pM;
sl@0
   272
			++pM;
sl@0
   273
			TInt bLen=eB-pB;
sl@0
   274
			if (bLen<len)
sl@0
   275
				return EFalse;
sl@0
   276
			if (aTextOp.Find(pB,bLen,pM,len)<0)
sl@0
   277
				return EFalse;
sl@0
   278
			pM+=len;
sl@0
   279
			pB+=bLen;
sl@0
   280
			}
sl@0
   281
			break;
sl@0
   282
		case EBeginning:
sl@0
   283
			{
sl@0
   284
			TInt len=*++pM;
sl@0
   285
			++pM;
sl@0
   286
			if (eB-pB<len)
sl@0
   287
				return EFalse;
sl@0
   288
			if (segment&EWild)
sl@0
   289
				{
sl@0
   290
				if (aTextOp.Match(pB,len,pM,len)<0)
sl@0
   291
					return EFalse;
sl@0
   292
				}
sl@0
   293
			else
sl@0
   294
				{
sl@0
   295
				if (aTextOp.Compare(pB,len,pM,len)!=0)
sl@0
   296
					return EFalse;
sl@0
   297
				}
sl@0
   298
			pM+=len;
sl@0
   299
			pB+=len;
sl@0
   300
			}
sl@0
   301
			break;
sl@0
   302
		case EMiddle:
sl@0
   303
			{
sl@0
   304
			TInt len=*++pM;
sl@0
   305
			++pM;
sl@0
   306
			TInt match=len;
sl@0
   307
			if (segment&EWild)
sl@0
   308
				match-=2;	// the number of characters to match
sl@0
   309
			TInt left=eB-pB;
sl@0
   310
			if (left<match)
sl@0
   311
				pB=UnderflowL(eB,left);
sl@0
   312
			for (;;)
sl@0
   313
				{
sl@0
   314
				TInt bLen=eB-pB;
sl@0
   315
				if (bLen<match)
sl@0
   316
					return EFalse;
sl@0
   317
				TInt pos=segment&EWild ? aTextOp.Match(pB,bLen,pM,len) : aTextOp.Find(pB,bLen,pM,len);
sl@0
   318
				if (pos>=0)
sl@0
   319
					{	// found it
sl@0
   320
					pB+=pos+match;
sl@0
   321
					break;
sl@0
   322
					}
sl@0
   323
				// not found, next chunk of data please.
sl@0
   324
				pB=UnderflowL(eB,match-1);
sl@0
   325
				}
sl@0
   326
			pM+=len;
sl@0
   327
			}
sl@0
   328
			break;
sl@0
   329
		case EEnd:	// match the last segment
sl@0
   330
			{
sl@0
   331
			TInt len=*++pM;
sl@0
   332
			++pM;
sl@0
   333
			TInt left=eB-pB;
sl@0
   334
			if (left<len)
sl@0
   335
				{
sl@0
   336
				pB=UnderflowL(eB,left);
sl@0
   337
				if (eB-pB<len)
sl@0
   338
					return EFalse;
sl@0
   339
				}
sl@0
   340
			while (eB-pB>len)
sl@0
   341
				pB=UnderflowL(eB,len);
sl@0
   342
			if (segment&EWild)
sl@0
   343
				{
sl@0
   344
				if (aTextOp.Match(pB,len,pM,len)<0)
sl@0
   345
					return EFalse;
sl@0
   346
				}
sl@0
   347
			else
sl@0
   348
				{
sl@0
   349
				if (aTextOp.Compare(pB,len,pM,len)!=0)
sl@0
   350
					return EFalse;
sl@0
   351
				}
sl@0
   352
			}
sl@0
   353
			return ETrue;
sl@0
   354
			}
sl@0
   355
		}
sl@0
   356
	}
sl@0
   357
sl@0
   358
// Class TDesMatcher
sl@0
   359
sl@0
   360
template <class T,class D>
sl@0
   361
inline TDesMatcher<T,D>::TDesMatcher(const D& aDes)
sl@0
   362
	{iEnd=(iPtr=aDes.Ptr())+aDes.Length();}
sl@0
   363
sl@0
   364
template <class T,class D>
sl@0
   365
const T* TDesMatcher<T,D>::UnderflowL(const T*& aEnd,TInt aRetain)
sl@0
   366
	{
sl@0
   367
	const T* ptr=iPtr-aRetain;
sl@0
   368
	aEnd=iPtr=iEnd;
sl@0
   369
	return ptr;
sl@0
   370
	}
sl@0
   371
sl@0
   372
// Class TStreamMatcher
sl@0
   373
sl@0
   374
template <class T,class D>
sl@0
   375
inline TStreamMatcher<T,D>::TStreamMatcher(MStreamBuf& aStreamBuf,TInt aLength)
sl@0
   376
	:iStream(&aStreamBuf),iRemain(aLength),iEnd(&iBuffer[EBufSize])
sl@0
   377
	{}
sl@0
   378
sl@0
   379
template <class T,class D>
sl@0
   380
const T* TStreamMatcher<T,D>::UnderflowL(const T*& aEnd,TInt aRetain)
sl@0
   381
	{
sl@0
   382
	if (iRemain==0)
sl@0
   383
		{	// no more stream data, don't move etc.
sl@0
   384
		aEnd=iEnd;
sl@0
   385
		return iEnd-aRetain;
sl@0
   386
		}
sl@0
   387
	else
sl@0
   388
		{
sl@0
   389
		T* rp=&iBuffer[0];
sl@0
   390
		if (aRetain)
sl@0
   391
			rp=TextCopy(rp,iEnd-aRetain,aRetain);
sl@0
   392
		TInt read=Min(EBufSize-aRetain,iRemain);
sl@0
   393
		iStream.ReadL(rp,read);
sl@0
   394
		iRemain-=read;
sl@0
   395
		aEnd=iEnd=rp+read; 
sl@0
   396
		return iBuffer;
sl@0
   397
		}
sl@0
   398
	}
sl@0
   399
sl@0
   400
// Class RSqlLiteral
sl@0
   401
sl@0
   402
void RSqlLiteral::ToPattern8L(TBool aEscape)
sl@0
   403
//
sl@0
   404
// Convert a Buf8 to an 8-bit pattern
sl@0
   405
//
sl@0
   406
	{
sl@0
   407
	__ASSERT(iType==EBuf8);
sl@0
   408
	HMatcherPattern8* pattern=HMatcherPattern8::NewL(Text8(),aEscape);
sl@0
   409
	User::Free(iVal.iAlloc);
sl@0
   410
	iVal.iAlloc=pattern;
sl@0
   411
	iType=EPattern8;
sl@0
   412
	}
sl@0
   413
sl@0
   414
void RSqlLiteral::ToPattern16L(TBool aEscape)
sl@0
   415
//
sl@0
   416
// Convert a Buf8 to an 8-bit pattern
sl@0
   417
//
sl@0
   418
	{
sl@0
   419
	__ASSERT(iType==EBuf16);
sl@0
   420
	HMatcherPattern16* pattern=HMatcherPattern16::NewL(Text16(),aEscape);
sl@0
   421
	User::Free(iVal.iAlloc);
sl@0
   422
	iVal.iAlloc=pattern;
sl@0
   423
	iType=EPattern16;
sl@0
   424
	}
sl@0
   425
sl@0
   426
sl@0
   427
// Class CSqlLikePredicate
sl@0
   428
sl@0
   429
LOCAL_C TInt MatchLength(const TText* aPtr,const TText* aEnd)
sl@0
   430
	{
sl@0
   431
	TInt match=0;
sl@0
   432
	while (aPtr<aEnd)
sl@0
   433
		{
sl@0
   434
		TText c=*aPtr++;
sl@0
   435
		if (c!=KMatchAny)
sl@0
   436
			{
sl@0
   437
			++match;
sl@0
   438
			if (c=='\'')
sl@0
   439
				++aPtr;
sl@0
   440
			}
sl@0
   441
		}
sl@0
   442
	return match;
sl@0
   443
	}
sl@0
   444
//Following code is for the implementation of limited-ESCAPE-clause 
sl@0
   445
LOCAL_C TInt MatchLengthEscape(const TText* aPtr,const TText* aEnd)
sl@0
   446
	{
sl@0
   447
	TInt match=0;
sl@0
   448
	while (aPtr<aEnd)
sl@0
   449
		{
sl@0
   450
		TText c=*aPtr++;
sl@0
   451
			{
sl@0
   452
			++match;
sl@0
   453
			if (c=='\'')
sl@0
   454
				++aPtr;
sl@0
   455
			}
sl@0
   456
		}
sl@0
   457
	return match;
sl@0
   458
	}
sl@0
   459
sl@0
   460
sl@0
   461
CSqlLikePredicate::CSqlLikePredicate(TType aType,const TDesC& aColumn,const RSqlLiteral& aPattern)
sl@0
   462
	: CSqlLiteralNode(aType,aColumn,aPattern)
sl@0
   463
	{
sl@0
   464
	__ASSERT(aType==ELike||aType==ENotLike);
sl@0
   465
	}
sl@0
   466
sl@0
   467
void CSqlLikePredicate::BindL(const RDbTableRow& aSource)
sl@0
   468
//
sl@0
   469
// Compile the Pattern for LongText columns and evaluate the match length
sl@0
   470
//
sl@0
   471
	{
sl@0
   472
	TInt matchsize;
sl@0
   473
	if(iIsEscape)//Following code is for the implementation of limited-ESCAPE-clause 
sl@0
   474
		{
sl@0
   475
		matchsize = MatchLengthEscape(Value().Ptr(),Value().End());				
sl@0
   476
		}
sl@0
   477
	else
sl@0
   478
		{
sl@0
   479
		matchsize = MatchLength(Value().Ptr(),Value().End());	
sl@0
   480
		}
sl@0
   481
	CSqlLiteralNode::BindL(aSource);
sl@0
   482
	switch (ColType())
sl@0
   483
		{
sl@0
   484
	default:	// type mismatch: should be caught by Literal::Bind
sl@0
   485
		__ASSERT(0);
sl@0
   486
	case EDbColText8:
sl@0
   487
		break;
sl@0
   488
	case EDbColLongText8:
sl@0
   489
		Value().ToPattern8L(iIsEscape);
sl@0
   490
		break;
sl@0
   491
	case EDbColLongText16:
sl@0
   492
		Value().ToPattern16L(iIsEscape);
sl@0
   493
		matchsize<<=1;
sl@0
   494
		break;
sl@0
   495
	case EDbColText16:
sl@0
   496
		matchsize<<=1;
sl@0
   497
		break;
sl@0
   498
		}
sl@0
   499
	iMatchSize=matchsize;
sl@0
   500
	}
sl@0
   501
sl@0
   502
TBool CSqlLikePredicate::EvaluateL(const TTextOps& aTextOp) const
sl@0
   503
	{
sl@0
   504
	__ASSERT(IsBound());
sl@0
   505
	TDbColType type=ColType();
sl@0
   506
	TDbColumnC column(Column());
sl@0
   507
	TInt match;
sl@0
   508
	if (TDbCol::IsLong(type))
sl@0
   509
		{
sl@0
   510
		const TDbBlob& blob=column.Blob();
sl@0
   511
		if (blob.Size()<iMatchSize)
sl@0
   512
			match=EFalse;
sl@0
   513
		else if (blob.IsInline())
sl@0
   514
			match=type==EDbColLongText8 ? 
sl@0
   515
				Value().Pattern8().MatchL(blob.PtrC8(),aTextOp) : 
sl@0
   516
				Value().Pattern16().MatchL(blob.PtrC16(),aTextOp);
sl@0
   517
		else
sl@0
   518
			{
sl@0
   519
			MStreamBuf& buf=StreamLC(blob);
sl@0
   520
			match=type==EDbColLongText8 ? 
sl@0
   521
				Value().Pattern8().MatchL(buf,blob.Size(),aTextOp) : 
sl@0
   522
				Value().Pattern16().MatchL(buf,blob.Size()>>1,aTextOp);
sl@0
   523
			CleanupStack::PopAndDestroy();
sl@0
   524
			}
sl@0
   525
		}
sl@0
   526
	else if (column.Size()<iMatchSize)
sl@0
   527
		match=EFalse;
sl@0
   528
	else
sl@0
   529
		{
sl@0
   530
			if(iIsEscape) //Following code is for the implementation of limited-ESCAPE-clause 
sl@0
   531
			{
sl@0
   532
			match=type==EDbColText8 ?
sl@0
   533
				aTextOp.Find(column.PtrC8(),Value().Text8()) :
sl@0
   534
				aTextOp.Find(column.PtrC16(),Value().Text16());
sl@0
   535
			}
sl@0
   536
			else
sl@0
   537
			{
sl@0
   538
			match=type==EDbColText8 ?
sl@0
   539
				aTextOp.Match(column.PtrC8(),Value().Text8()) :
sl@0
   540
				aTextOp.Match(column.PtrC16(),Value().Text16());
sl@0
   541
				
sl@0
   542
			}
sl@0
   543
		
sl@0
   544
		match=match>=0;
sl@0
   545
		}
sl@0
   546
	return NodeType()==ELike ? match : !match;
sl@0
   547
	}