os/persistentdata/persistentstorage/dbms/ustor/US_COMP.CPP
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
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".
     7 //
     8 // Initial Contributors:
     9 // Nokia Corporation - initial contribution.
    10 //
    11 // Contributors:
    12 //
    13 // Description:
    14 // Store database compression code
    15 // 
    16 //
    17 
    18 #include "US_STD.H"
    19 #include <s32mem.h>
    20 
    21 #ifdef __WINS__
    22 #define __EXTRA_DEFLATE
    23 #endif
    24 
    25 // deflation constants
    26 const TInt KDeflateMinLength=3;
    27 const TInt KDeflateMaxLength=258;
    28 const TInt KDeflateMaxDistance=4096;
    29 const TInt KDeflateDistCodeBase=0x200;
    30 // huffman coding/decoding
    31 const TInt KHuffMaxCodeLength=25;
    32 const TInt KHuffTerminate=1;
    33 const TUint KBitsEmpty=0x80000000u;
    34 const TUint KBitsInit=KBitsEmpty>>1;
    35 const TUint KBitsFull=KBitsEmpty>>8;
    36 const TUint KBitsEOF=KBitsEmpty>>9;
    37 const TUint KBitsNext=0x80u;
    38 // encoding storage
    39 const TInt KDeflateMetaCodes=26;
    40 // hashing
    41 const TUint KDeflateHashMultiplier=0xAC4B9B19u;
    42 const TInt KDeflateHashShift=24;
    43 
    44 class Huffman
    45 	{
    46 public:
    47 	static void EncodingL(TUint32* aEncoding,TInt aCodes);
    48 	static void Decoding(TUint32* aDecoding,TInt aCodes,TInt aBase=0);
    49 private:
    50 	typedef TUint16 THuff;
    51 	enum {KLeaf=0x8000};
    52 	struct TNode
    53 		{
    54 		THuff iLeft;
    55 		THuff iRight;
    56 		};
    57 	struct TLeaf
    58 		{
    59 		TUint iCount;
    60 		THuff iVal;
    61 		};
    62 private:
    63 	static void Lengths(TUint32* aLengths,const TNode* aNodes,TInt aNode,TInt aLen);
    64 	static TUint32* SubTree(TUint32* aPtr,const TUint32* aValue,TUint32** aLevel);
    65 	};
    66 
    67 class THuffEncoder
    68 	{
    69 public:
    70 	THuffEncoder(RWriteStream& aStream);
    71 //
    72 	void EncodeL(TUint aCode,TInt aLength);
    73 	void EncodeL(TUint32 aHuffCode);
    74 	void CompleteL();
    75 private:
    76 	enum {EBufSize=0x100};
    77 private:
    78 	TUint8 iBuf[EBufSize];
    79 	RWriteStream& iStream;
    80 	TUint32 iCode;		// code in production
    81 	TInt iBits;
    82 	TUint8* iWrite;
    83 	};
    84 
    85 class HDeflateHash
    86 	{
    87 public:
    88 	inline static HDeflateHash& NewLC(TInt aLinks);
    89 //
    90 	inline TInt First(const TUint8* aPtr,TInt aPos);
    91 	inline TInt Next(TInt aPos,TInt aOffset) const;
    92 private:
    93 	inline HDeflateHash();
    94 	inline static TInt Hash(const TUint8* aPtr);
    95 private:
    96 	typedef TUint16 TOffset;
    97 private:
    98 	TInt iHash[256];
    99 	TOffset iOffset[1];	// or more
   100 	};
   101 
   102 class MDeflater
   103 	{
   104 public:
   105 	void DeflateL(const TUint8* aBase,TInt aLength);
   106 private:
   107 	const TUint8* DoDeflateL(const TUint8* aBase,const TUint8* aEnd,HDeflateHash& aHash);
   108 	static TInt Match(const TUint8* aPtr,const TUint8* aEnd,TInt aPos,HDeflateHash& aHas);
   109 	void SegmentL(TInt aLength,TInt aDistance);
   110 	virtual void LitLenL(TInt aCode) =0;
   111 	virtual void OffsetL(TInt aCode) =0;
   112 	virtual void ExtraL(TInt aLen,TUint aBits) =0;
   113 	};
   114 
   115 class TInflater
   116 	{
   117 public:
   118 	TInflater(const TUint8* aIn,const CDbStoreCompression::TEncoding& aDecoding);
   119 	TInt Inflate();
   120 	inline const TUint8* Ptr() const;
   121 	inline static TInt BufferSize();
   122 private:
   123 	const TUint8* iIn;
   124 	TUint iBits;
   125 	const TUint8* iRptr;			// partial segment
   126 	TInt iLen;
   127 	const TUint32* iLitLenTree;
   128 	const TUint32* iDistTree;
   129 	TUint8 iOut[KDeflateMaxDistance];	// circular buffer for distance matches
   130 	};
   131 
   132 NONSHARABLE_CLASS(TDeflateStats) : public MDeflater
   133 	{
   134 public:
   135 	inline TDeflateStats(CDbStoreCompression::TEncoding& aEncoding);
   136 private:
   137 // from MDeflater
   138 	void LitLenL(TInt aCode);
   139 	void OffsetL(TInt aCode);
   140 	void ExtraL(TInt aLen,TUint aBits);
   141 private:
   142 	CDbStoreCompression::TEncoding& iEncoding;
   143 	};
   144 
   145 NONSHARABLE_CLASS(TDeflater) : public MDeflater
   146 	{
   147 public:
   148 	inline TDeflater(THuffEncoder& aEncoder,const CDbStoreCompression::TEncoding& aEncoding);
   149 private:
   150 // from MDeflater
   151 	void LitLenL(TInt aCode);
   152 	void OffsetL(TInt aCode);
   153 	void ExtraL(TInt aLen,TUint aBits);
   154 private:
   155 	THuffEncoder& iEncoder;
   156 	const CDbStoreCompression::TEncoding& iEncoding;
   157 	};
   158 
   159 NONSHARABLE_CLASS(HDeflateBuf) : public TBufBuf
   160 	{
   161 public:
   162 	enum TMode {EAnalysis,EDeflate};	// mirror CDbStoreCompression enum
   163 public:
   164 	static HDeflateBuf* NewL(MStreamBuf* aHost,CDbStoreCompression::TEncoding& aEncoding,TMode aMode);
   165 private:
   166 	inline HDeflateBuf(MStreamBuf* aHost,CDbStoreCompression::TEncoding& aEncoding,CBufBase* aBuf,TMode aMode);
   167 	virtual inline ~HDeflateBuf();
   168 // from MStreamBuf
   169 	void DoSynchL();
   170 	void DoRelease();
   171 private:
   172 	RWriteStream iHost;
   173 	CDbStoreCompression::TEncoding& iEncoding;
   174 	CBufBase* iBuf;
   175 	TMode iMode;
   176 	};
   177 
   178 NONSHARABLE_CLASS(HInflateBuf) : public TBufBuf
   179 	{
   180 public:
   181 	static HInflateBuf* NewL(MStreamBuf* aHost,const CDbStoreCompression::TEncoding& aEncoding);
   182 private:
   183 	inline HInflateBuf(CBufBase* aBuf);
   184 	virtual inline ~HInflateBuf();
   185 // from MStreamBuf
   186 	void DoRelease();
   187 private:
   188 	CBufBase* iBuf;
   189 	};
   190 
   191 NONSHARABLE_CLASS(CDbStoreTable::CCompressor) : public CBase, public CCluster::MAlter
   192 	{
   193 public:
   194 	inline CCompressor();
   195 	~CCompressor();
   196 	void ProcessL(CDbStoreTable* aTable);
   197 private:
   198 	TUint8* AlterRecordL(TUint8* aWPtr,const TUint8* aRPtr,TInt aLength);
   199 private:
   200 	CDbStoreTable* iTable;
   201 	RDbRow iRow;
   202 	};
   203 
   204 // Class Huffman
   205 //
   206 // This class builds a huffman encoding from a frequency table and builds
   207 // a decoding tree from a code-lengths table
   208 //
   209 // the encoding generated is based on the rule that given two symbols s1 and s2, with 
   210 // code length l1 and l2, and huffman codes h1 and h2:
   211 //
   212 // if l1<l2 then h1<h2 when compared lexicographically
   213 // if l1==l2 and s1<s2 then h1<h2 ditto
   214 //
   215 // This allows the encoding to be stored compactly as a table of code lengths
   216 
   217 //
   218 // recursive function to calculate the code lengths from the node tree
   219 //
   220 void Huffman::Lengths(TUint32* aLengths,const TNode* aNodes,TInt aNode,TInt aLen)
   221 	{
   222 	__ASSERT(aLen<KHuffMaxCodeLength);
   223 	++aLen;
   224 	const TNode& node=aNodes[aNode];
   225 	if (node.iLeft&KLeaf)
   226 		aLengths[node.iLeft&~KLeaf]=aLen;
   227 	else
   228 		Lengths(aLengths,aNodes,node.iLeft,aLen);
   229 	if (node.iRight&KLeaf)
   230 		aLengths[node.iRight&~KLeaf]=aLen;
   231 	else
   232 		Lengths(aLengths,aNodes,node.iRight,aLen);
   233 	}
   234 
   235 //
   236 // write the subtree below aPtr and return the head
   237 //
   238 TUint32* Huffman::SubTree(TUint32* aPtr,const TUint32* aValue,TUint32** aLevel)
   239 	{
   240 	TUint32* l=*aLevel++;
   241 	if (l>aValue)
   242 		{
   243 		TUint32* sub1=SubTree(aPtr,aValue,aLevel);	// 0-tree first
   244 		aPtr=SubTree(sub1,aValue-(aPtr-sub1)-1,aLevel);			// 1-tree
   245 		TInt branch=(TUint8*)sub1-(TUint8*)aPtr;
   246 		*--aPtr=branch;
   247 		}
   248 	else if (l==aValue)
   249 		{
   250 		TUint term0=*aValue--;						// 0-term
   251 		aPtr=SubTree(aPtr,aValue,aLevel);			// 1-tree
   252 		*--aPtr=term0>>16;
   253 		}
   254 	else	// l<iNext
   255 		{
   256 		TUint term0=*aValue--;						// 0-term
   257 		TUint term1=*aValue--;
   258 		*--aPtr=(term1>>16<<16)|(term0>>16);
   259 		}
   260 	return aPtr;
   261 	}
   262 
   263 //
   264 // Build a huffman encoding table from a symbol frequency table
   265 // aTable contains frequency data on input for aCodes symbols
   266 // aTable contains the huffman encoding on output
   267 //
   268 void Huffman::EncodingL(TUint32* aTable,TInt aCodes)
   269 	{
   270 //
   271 // step 1. Sort the values into decreasing order of frequency
   272 //
   273 	TLeaf* leaves=new(ELeave) TLeaf[aCodes];
   274 	CleanupArrayDeletePushL(leaves);
   275 	TInt lCount=0;
   276 	TInt ii;
   277 	for (ii=0;ii<aCodes;++ii)
   278 		{
   279 		TUint c=aTable[ii];
   280 		if (c==0)
   281 			continue;	// no coding for ii
   282 		TInt jj;
   283 		for (jj=0;jj<lCount;++jj)
   284 			{
   285 			if (leaves[jj].iCount<=c)
   286 				break;
   287 			}
   288 		Mem::Move(leaves+jj+1,leaves+jj,sizeof(TLeaf)*(lCount-jj));
   289 		leaves[jj].iCount=c;
   290 		leaves[jj].iVal=THuff(ii|KLeaf);
   291 		lCount++;
   292 		}
   293 //
   294 // Huffman algorithm: pair off least frequent nodes and reorder
   295 // result is the code lengths in aTable[]
   296 //
   297 	if (lCount==1)	// special case for a single value (always encode as "0")
   298 		aTable[leaves[0].iVal&~KLeaf]=1;
   299 	else if (lCount>1)
   300 		{	//	don't encode for empty coding: leaves in order now
   301 		TInt max=0;
   302 		TNode* nodes=new(ELeave) TNode[lCount-1];
   303 		while (--lCount>0)
   304 			{
   305 			TNode& node=nodes[max];
   306 			node.iLeft=leaves[lCount-1].iVal;
   307 			node.iRight=leaves[lCount].iVal;
   308 			// re-order the leaves now to reflect new combined frequency
   309 			TUint c=leaves[lCount-1].iCount+leaves[lCount].iCount;
   310 			TInt jj=lCount;
   311 			while (--jj>0)
   312 				{
   313 				if (leaves[jj-1].iCount>=c)
   314 					break;
   315 				}
   316 			Mem::Move(leaves+jj+1,leaves+jj,sizeof(TLeaf)*(lCount-1-jj));
   317 			// update new leaf
   318 			leaves[jj].iCount=c;
   319 			leaves[jj].iVal=THuff(max);
   320 			max++;
   321 			}
   322 		Lengths(aTable,nodes,leaves[0].iVal,0);
   323 		delete[] nodes;
   324 		}
   325 	CleanupStack::PopAndDestroy();		// leaves
   326 //
   327 // step 3: Generate the coding based on the code lengths
   328 //
   329 	TInt lenCount[KHuffMaxCodeLength];
   330 	Mem::FillZ(lenCount,sizeof(lenCount));
   331 
   332 	for (ii=aCodes;--ii>=0;)
   333 		{
   334 		TInt len=aTable[ii]-1;
   335 		if (len>=0)
   336 			++lenCount[len];
   337 		}
   338 
   339 	TUint nextCode[KHuffMaxCodeLength];
   340 	TUint code=0;
   341 	for (ii=0;ii<KHuffMaxCodeLength;++ii)
   342 		{
   343 		nextCode[ii]=code;
   344 		code=(code+lenCount[ii])<<1;
   345 		}
   346 
   347 	for (ii=0;ii<aCodes;++ii)
   348 		{
   349 		TInt len=aTable[ii];
   350 		if (len)
   351 			{
   352 			aTable[ii] = (nextCode[len-1]<<(KHuffMaxCodeLength-len))|(len<<KHuffMaxCodeLength);
   353 			++nextCode[len-1];
   354 			}
   355 		}
   356 	}
   357 
   358 //
   359 // generate the decoding tree from the huffman code length data
   360 // output alphabet is [aBase,aBase+aCodes)
   361 //
   362 void Huffman::Decoding(TUint32* aDecoding,TInt aCodes,TInt aBase)
   363 	{
   364 	TInt counts[KHuffMaxCodeLength];
   365 	Mem::FillZ(counts,sizeof(counts));
   366 	TInt codes=0;
   367 	TInt ii=aCodes;
   368 	while (--ii>=0)
   369 		{
   370 		TUint len=aDecoding[ii];
   371 		__ASSERT(len<=(TUint)KHuffMaxCodeLength);
   372 		if (len)
   373 			{
   374 			++counts[len-1];
   375 			++codes;
   376 			}
   377 		}
   378 //
   379 	TUint32* level[KHuffMaxCodeLength];
   380 	TUint32* lit=aDecoding+codes;
   381 	for (ii=0;ii<KHuffMaxCodeLength;++ii)
   382 		{
   383 		level[ii]=lit;
   384 		lit-=counts[ii];
   385 		}
   386 	aBase=(aBase<<17)+(KHuffTerminate<<16);
   387 	for (ii=0;ii<aCodes;++ii)
   388 		{
   389 		TUint len=TUint8(aDecoding[ii]);
   390 		if (len)
   391 			*--level[len-1]|=(ii<<17)+aBase;
   392 		}
   393 	if (codes==1)	// codes==1 special case: tree is not complete
   394 		*aDecoding>>=16;	// 0-terminate at root
   395 	else if (codes>1)
   396 		{
   397 		__DEBUG(TUint32* p=) SubTree(aDecoding+codes-1,aDecoding+codes-1,level);
   398 		__ASSERT(p==aDecoding);
   399 		}
   400 	}
   401 
   402 // Class HDeflateHash
   403 
   404 inline HDeflateHash::HDeflateHash()
   405 	{TInt* p=iHash+256;do *--p=-KDeflateMaxDistance-1; while (p>iHash);}
   406 
   407 inline HDeflateHash& HDeflateHash::NewLC(TInt aLinks)
   408 	{
   409 	__ASSERT(!(KDeflateMaxDistance&(KDeflateMaxDistance-1)));	// ensure power of two
   410 	return *new(User::AllocLC(_FOFF(HDeflateHash,iOffset[Min(aLinks,KDeflateMaxDistance)]))) HDeflateHash;
   411 	}
   412 
   413 inline TInt HDeflateHash::Hash(const TUint8* aPtr)
   414 	{
   415 	TUint x=aPtr[0]|(aPtr[1]<<8)|(aPtr[2]<<16);
   416 	return (x*KDeflateHashMultiplier)>>KDeflateHashShift;
   417 	}
   418 
   419 inline TInt HDeflateHash::First(const TUint8* aPtr,TInt aPos)
   420 	{
   421 	TInt h=Hash(aPtr);
   422 	TInt offset=Min(aPos-iHash[h],KDeflateMaxDistance<<1);
   423 	iHash[h]=aPos;
   424 	iOffset[aPos&(KDeflateMaxDistance-1)]=TOffset(offset);
   425 	return offset;
   426 	}
   427 
   428 inline TInt HDeflateHash::Next(TInt aPos,TInt aOffset) const
   429 	{return aOffset+iOffset[(aPos-aOffset)&(KDeflateMaxDistance-1)];}
   430 
   431 
   432 // Class TDeflater
   433 //
   434 // generic deflation algorithm, can do either statistics and the encoder
   435 
   436 TInt MDeflater::Match(const TUint8* aPtr,const TUint8* aEnd,TInt aPos,HDeflateHash& aHash)
   437 	{
   438 	TInt offset=aHash.First(aPtr,aPos);
   439 	if (offset>KDeflateMaxDistance)
   440 		return 0;
   441 	TInt match=0;
   442 	aEnd=Min(aEnd,aPtr+KDeflateMaxLength);
   443 	TUint8 c=*aPtr;
   444 	do
   445 		{
   446 		const TUint8* p=aPtr-offset;
   447 		if (p[match>>16]==c)
   448 			{	// might be a better match
   449 			const TUint8* m=aPtr;
   450 			for (;;)
   451 				{
   452 				if (*p++!=*m++)
   453 					break;
   454 				if (m<aEnd)
   455 					continue;
   456 				return ((m-aPtr)<<16)|offset;
   457 				}
   458 			TInt l=m-aPtr-1;
   459 			if (l>match>>16)
   460 				{
   461 				match=(l<<16)|offset;
   462 				c=m[-1];
   463 				}
   464 			}
   465 		offset=aHash.Next(aPos,offset);
   466 		} while (offset<=KDeflateMaxDistance);
   467 	return match;
   468 	}
   469 
   470 //
   471 // Apply the deflation algorithm to the data [aBase,aEnd)
   472 // Return a pointer after the last byte that was deflated (which may not be aEnd)
   473 //
   474 const TUint8* MDeflater::DoDeflateL(const TUint8* aBase,const TUint8* aEnd,HDeflateHash& aHash)
   475 	{
   476 	__ASSERT(aEnd-aBase>KDeflateMinLength);
   477 //
   478 	const TUint8* ptr=aBase;
   479 #ifdef __EXTRA_DEFLATE
   480 	TInt prev=0;		// the previous deflation match
   481 #endif
   482 	do
   483 		{
   484 		TInt match=Match(ptr,aEnd,ptr-aBase,aHash);
   485 #ifdef __EXTRA_DEFLATE
   486 // Extra deflation applies two optimisations which double the time taken
   487 // 1. If we have a match at p, then test for a better match at p+1 before using it
   488 // 2. When we have a match, add the hash links for all the data which will be skipped 
   489 		if (match>>16 < prev>>16)
   490 			{	// use the previous match--it was better
   491 			TInt len=prev>>16;
   492 			SegmentL(len,prev-(len<<16));
   493 			// fill in missing hash entries for better compression
   494 			const TUint8* e=ptr+len-2;
   495 			do
   496 				{
   497 				++ptr;
   498 				aHash.First(ptr,ptr-aBase);
   499 				} while (ptr<e);
   500 			prev=0;
   501 			}
   502 		else if (match<=(KDeflateMinLength<<16))
   503 			LitLenL(*ptr);			// no deflation match here
   504 		else
   505 			{	// save this match and test the next position
   506 			if (prev)	// we had a match at ptr-1, but this is better
   507 				LitLenL(ptr[-1]);
   508 			prev=match;
   509 			}
   510 		++ptr;
   511 #else
   512 // Basic deflation will store any match found, and not update the hash links for the
   513 // data which is skipped
   514 		if (match<=(KDeflateMinLength<<16))		// no match
   515 			LitLenL(*ptr++);
   516 		else
   517 			{	// store the match
   518 			TInt len=match>>16;
   519 			SegmentL(len,match-(len<<16));
   520 			ptr+=len;
   521 			}
   522 #endif
   523 		} while (ptr+KDeflateMinLength-1<aEnd);
   524 #ifdef __EXTRA_DEFLATE
   525 	if (prev)
   526 		{		// emit the stored match
   527 		TInt len=prev>>16;
   528 		SegmentL(len,prev-(len<<16));
   529 		ptr+=len-1;
   530 		}
   531 #endif
   532 	return ptr;
   533 	}
   534 
   535 //
   536 // The generic deflation algorithm
   537 //
   538 void MDeflater::DeflateL(const TUint8* aBase,TInt aLength)
   539 	{
   540 	const TUint8* end=aBase+aLength;
   541 	if (aLength>KDeflateMinLength)
   542 		{	// deflation kicks in if there is enough data
   543 		HDeflateHash& hash=HDeflateHash::NewLC(aLength);
   544 		aBase=DoDeflateL(aBase,end,hash);
   545 		CleanupStack::PopAndDestroy();
   546 		}
   547 	while (aBase<end)					// emit remaining bytes
   548 		LitLenL(*aBase++);
   549 	LitLenL(CDbStoreCompression::TEncoding::EEos);	// eos marker
   550 	}
   551 
   552 //
   553 // Turn a (length,offset) pair into the deflation codes+extra bits before calling
   554 // the specific LitLen(), Offset() and Extra() functions.
   555 //
   556 void MDeflater::SegmentL(TInt aLength,TInt aDistance)
   557 	{
   558 	__ASSERT(aLength>=KDeflateMinLength && aLength<=KDeflateMaxLength);
   559 	aLength-=KDeflateMinLength;
   560 	TInt extralen=0;
   561 	TUint len=aLength;
   562 	while (len>=8)
   563 		{
   564 		++extralen;
   565 		len>>=1;
   566 		}
   567 	__ASSERT((extralen<<2)+len<CDbStoreCompression::TEncoding::ELengths);
   568 	LitLenL((extralen<<2)+len+CDbStoreCompression::TEncoding::ELiterals);
   569 	if (extralen)
   570 		ExtraL(extralen,TUint(aLength)<<(32-extralen));
   571 //
   572 	__ASSERT(aDistance>0 && aDistance<=KDeflateMaxDistance);
   573 	aDistance--;
   574 	extralen=0;
   575 	TUint dist=aDistance;
   576 	while (dist>=8)
   577 		{
   578 		++extralen;
   579 		dist>>=1;
   580 		}
   581 	__ASSERT((extralen<<2)+dist<CDbStoreCompression::TEncoding::EDistances);
   582 	OffsetL((extralen<<2)+dist);
   583 	if (extralen)
   584 		ExtraL(extralen,TUint(aDistance)<<(32-extralen));
   585 	}
   586 
   587 // Class TDeflateStats
   588 //
   589 // This class analyses the data stream to generate the frequency tables 
   590 // for the deflation algorithm
   591 
   592 inline TDeflateStats::TDeflateStats(CDbStoreCompression::TEncoding& aEncoding)
   593 	:iEncoding(aEncoding)
   594 	{}
   595 
   596 void TDeflateStats::LitLenL(TInt aCode)
   597 	{
   598 	++iEncoding.iLitLen[aCode];
   599 	}
   600 
   601 void TDeflateStats::OffsetL(TInt aCode)
   602 	{
   603 	++iEncoding.iDistance[aCode];
   604 	}
   605 
   606 void TDeflateStats::ExtraL(TInt,TUint)
   607 	{}
   608 
   609 // Class THuffEncoder
   610 //
   611 // This class generates the byte stream of huffman codes, writing them out to the stream
   612 
   613 THuffEncoder::THuffEncoder(RWriteStream& aStream)
   614 	:iStream(aStream),iCode(0),iBits(-8),iWrite(iBuf)
   615 	{}
   616 
   617 //
   618 // Store a huffman code generated by Huffman::EncodingL()
   619 //
   620 void THuffEncoder::EncodeL(TUint32 aHuffCode)
   621 	{
   622 	EncodeL(aHuffCode<<(32-KHuffMaxCodeLength),aHuffCode>>KHuffMaxCodeLength);
   623 	}
   624 
   625 //
   626 // Store aLength bits from the most significant bits of aCode
   627 //
   628 void THuffEncoder::EncodeL(TUint aCode,TInt aLength)
   629 	{
   630 	TInt bits=iBits;
   631 	TUint code=iCode|(aCode>>(bits+8));
   632 	bits+=aLength;
   633 	if (bits>=0)
   634 		{
   635 		TUint8* write=iWrite;
   636 		do
   637 			{
   638 			if (write-EBufSize==iBuf)
   639 				{
   640 				iStream.WriteL(iBuf,EBufSize);
   641 				write=iBuf;
   642 				}
   643 			*write++=TUint8(code>>24);
   644 			code<<=8;
   645 			bits-=8;
   646 			} while (bits>=0);
   647 		iWrite=write;
   648 		}
   649 	iCode=code;
   650 	iBits=bits;
   651 	}
   652 
   653 //
   654 // Terminate the huffman coding. The longest code is always 1111111111
   655 //
   656 void THuffEncoder::CompleteL()
   657 	{
   658 	if (iBits>-8)
   659 		EncodeL(0xffffffffu,-iBits);
   660 	if (iWrite>iBuf)
   661 		iStream.WriteL(iBuf,iWrite-iBuf);
   662 	}
   663 
   664 // Class TDeflater
   665 //
   666 // Extends MDeflater to provide huffman encoding of the output
   667 
   668 //
   669 // construct for encoding
   670 //
   671 inline TDeflater::TDeflater(THuffEncoder& aEncoder,const CDbStoreCompression::TEncoding& aEncoding)
   672 	:iEncoder(aEncoder),iEncoding(aEncoding)
   673 	{}
   674 
   675 void TDeflater::LitLenL(TInt aCode)
   676 	{
   677 	iEncoder.EncodeL(iEncoding.iLitLen[aCode]);
   678 	}
   679 
   680 void TDeflater::OffsetL(TInt aCode)
   681 	{
   682 	iEncoder.EncodeL(iEncoding.iDistance[aCode]);
   683 	}
   684 
   685 void TDeflater::ExtraL(TInt aLen,TUint aBits)
   686 	{
   687 	iEncoder.EncodeL(aBits,aLen);
   688 	}
   689 
   690 // Class TInflater
   691 //
   692 // The inflation algorithm, complete with huffman decoding
   693 
   694 TInflater::TInflater(const TUint8* aIn,const CDbStoreCompression::TEncoding& aEncoding)
   695 	:iIn(aIn),iBits(KBitsInit),iLen(0),iLitLenTree(aEncoding.iLitLen),iDistTree(aEncoding.iDistance)
   696 	{}
   697 
   698 //
   699 // consume all data lag in the history buffer, then decode to fill up the output buffer
   700 //
   701 TInt TInflater::Inflate()
   702 	{
   703 // empty the history buffer into the output
   704 	const TUint8* data=iIn;
   705 	TUint bits=iBits;
   706 	const TUint8* from=iRptr;
   707 	TInt len=iLen;
   708 	TUint8* out=iOut;
   709 	TUint8* const end=out+KDeflateMaxDistance;
   710 	const TUint32* node;
   711 	if (len)
   712 		goto useHistory;
   713 //
   714 	if (bits&KBitsEOF)
   715 		return 0;
   716 //
   717 	node=iLitLenTree;
   718 	while (out<end)
   719 		{
   720 		// get a huffman code
   721 		{
   722 		TUint huff;
   723 		for (;;)
   724 			{
   725 			huff=*node++;
   726 			if ((bits<<=1)&KBitsEmpty)
   727 				bits=*data++|KBitsFull;
   728 			if (bits&KBitsNext)
   729 				{
   730 				if (huff&(KHuffTerminate<<16))
   731 					break;
   732 				}
   733 			else
   734 				{
   735 				if (huff&KHuffTerminate)
   736 					{
   737 					huff<<=16;
   738 					break;
   739 					}
   740 				else
   741 					node=PtrAdd(node,huff);
   742 				}
   743 			}
   744 		TInt val=TInt(huff>>17)-CDbStoreCompression::TEncoding::ELiterals;
   745 		if (val<0)
   746 			{
   747 			*out++=TUint8(val);
   748 			node=iLitLenTree;
   749 			continue;			// another literal/length combo
   750 			}
   751 		if (val==CDbStoreCompression::TEncoding::EEos-CDbStoreCompression::TEncoding::ELiterals)
   752 			{	// eos marker. we're done
   753 			bits=KBitsEOF;
   754 			break;
   755 			}
   756 		// get the extra bits for the code
   757 		TInt code=val&0xff;
   758 		if (code>=8)
   759 			{	// xtra bits
   760 			TInt xtra=(code>>2)-1;
   761 			code-=xtra<<2;
   762 			do
   763 				{
   764 				if ((bits<<=1)&KBitsEmpty)
   765 					bits=*data++|KBitsFull;
   766 				code<<=1;
   767 				if (bits&KBitsNext)
   768 					code|=1;
   769 				} while (--xtra!=0);
   770 			}
   771 		if (val<KDeflateDistCodeBase-CDbStoreCompression::TEncoding::ELiterals)
   772 			{	// length code... get the code
   773 			len=code+KDeflateMinLength;
   774 			__ASSERT(len<=KDeflateMaxLength);
   775 			node=iDistTree;
   776 			continue;			// read the huffman code
   777 			}
   778 		// distance code
   779 		__ASSERT(code<KDeflateMaxDistance);
   780 		from=out-(code+1);
   781 		if (from+KDeflateMaxDistance<end)
   782 			from+=KDeflateMaxDistance;
   783 		}
   784 useHistory:
   785 		TInt tfr=Min(end-out,len);
   786 		len-=tfr;
   787 		do
   788 			{
   789 			*out++=*from++;
   790 			if (from==end)
   791 				from-=KDeflateMaxDistance;
   792 			} while (--tfr!=0);
   793 		node=iLitLenTree;
   794 		};
   795 	iIn=data;
   796 	iBits=bits;
   797 	iRptr=from;
   798 	iLen=len;
   799 	return out-iOut;
   800 	}
   801 
   802 inline const TUint8* TInflater::Ptr() const
   803 	{return iOut;}
   804 inline TInt TInflater::BufferSize()
   805 	{return KDeflateMaxDistance;}
   806 
   807 // Class HDeflateBuf
   808 //
   809 // This stream buffer applies the analysis or deflation and huffman coding
   810 // on the entire stream data when it is committed
   811 
   812 inline HDeflateBuf::HDeflateBuf(MStreamBuf* aHost,CDbStoreCompression::TEncoding& aEncoding,CBufBase* aBuf,TMode aMode)
   813 	:iHost(aHost),iEncoding(aEncoding),iBuf(aBuf),iMode(aMode)
   814 	{Set(*aBuf,0);}
   815 
   816 HDeflateBuf* HDeflateBuf::NewL(MStreamBuf* aHost,CDbStoreCompression::TEncoding& aEncoding,TMode aMode)
   817 	{
   818 	CBufBase* buf=CBufFlat::NewL(512);
   819 	CleanupStack::PushL(buf);
   820 	HDeflateBuf* self=new(ELeave) HDeflateBuf(aHost,aEncoding,buf,aMode);
   821 	CleanupStack::Pop();
   822 	return self;
   823 	}
   824 
   825 inline HDeflateBuf::~HDeflateBuf()
   826 	{delete iBuf;iHost.Release();}
   827 
   828 void HDeflateBuf::DoRelease()
   829 	{
   830 	delete this;
   831 	}
   832 
   833 //
   834 // This is where it all happens
   835 //
   836 void HDeflateBuf::DoSynchL()
   837 	{
   838 	if (iMode==EAnalysis)
   839 		{
   840 		TDeflateStats deflater(iEncoding);
   841 		deflater.DeflateL(iBuf->Ptr(0).Ptr(),iBuf->Size());
   842 		}
   843 	else
   844 		{
   845 		THuffEncoder encoder(iHost);
   846 		TDeflater deflater(encoder,iEncoding);
   847 		deflater.DeflateL(iBuf->Ptr(0).Ptr(),iBuf->Size());
   848 		encoder.CompleteL();
   849 		iHost.CommitL();
   850 		}
   851 	}
   852 
   853 // Class HInflateBuf
   854 //
   855 // Inflate the input stream. This is not a filter, it reads all the input, inflates it and
   856 // keeps it in a memory buffer.
   857 
   858 const TInt KInflateBufSize=0x800;	// 2K
   859 
   860 HInflateBuf::HInflateBuf(CBufBase* aBuf)
   861 	:iBuf(aBuf)
   862 	{
   863 	Set(*aBuf,0,ERead);
   864 	}
   865 
   866 inline HInflateBuf::~HInflateBuf()
   867 	{delete iBuf;}
   868 
   869 void HInflateBuf::DoRelease()
   870 	{
   871 	delete this;
   872 	}
   873 
   874 HInflateBuf* HInflateBuf::NewL(MStreamBuf* aHost,const CDbStoreCompression::TEncoding& aEncoding)
   875 	{
   876 	CBufFlat* host=CBufFlat::NewL(256);
   877 	CleanupStack::PushL(host);
   878 	TUint8 buffer[KInflateBufSize];
   879 	for (;;)
   880 		{
   881 		TInt len=aHost->ReadL(buffer,KInflateBufSize);
   882 		if (len)
   883 			host->InsertL(host->Size(),buffer,len);
   884 		if (len<KInflateBufSize)
   885 			break;
   886 		}
   887 	CBufSeg* out=CBufSeg::NewL(256);
   888 	CleanupStack::PushL(out);
   889 	TInflater* inflater=new(ELeave) TInflater(host->Ptr(0).Ptr(),aEncoding);
   890 	CleanupStack::PushL(inflater);
   891 	for (;;)
   892 		{
   893 		TInt len=inflater->Inflate();
   894 		if (len)
   895 			out->InsertL(out->Size(),inflater->Ptr(),len);
   896 		if (len<inflater->BufferSize())
   897 			break;
   898 		}
   899 	HInflateBuf* buf=new(ELeave) HInflateBuf(out);
   900 	CleanupStack::PopAndDestroy();	// inflater
   901 	CleanupStack::Pop();			// out
   902 	CleanupStack::PopAndDestroy();	// host
   903 	aHost->Release();				// don't need this anymore
   904 	return buf;
   905 	}
   906 
   907 // Class CDbStoreTable::Compressor
   908 //
   909 // This class processes an entire table for analysis or compression, using the
   910 // CDbStoreRecords::AlterL() functionality and call back to ensure that all clusters
   911 // and BLOBs are read and written.
   912 
   913 inline CDbStoreTable::CCompressor::CCompressor()
   914 	{}
   915 
   916 CDbStoreTable::CCompressor::~CCompressor()
   917 	{
   918 	if (iTable)
   919 		iTable->Close();
   920 	iRow.Close();
   921 	}
   922 
   923 //
   924 // Walk through every cluster in the table
   925 //
   926 void CDbStoreTable::CCompressor::ProcessL(CDbStoreTable* aTable)
   927 	{
   928 	iTable=aTable;
   929 	CDbStoreRecords& rec=aTable->StoreRecordsL();
   930 	for (TClusterId cluster=rec.Head();cluster!=KNullClusterId;cluster=rec.AlterL(cluster,*this))
   931 		;
   932 	}
   933 
   934 //
   935 // Compress every blob, and transfer the record from aRPtr to aWPtr
   936 //
   937 TUint8* CDbStoreTable::CCompressor::AlterRecordL(TUint8* aWPtr,const TUint8* aRPtr,TInt aLength)
   938 	{
   939 	if (iTable->Def().Columns().HasLongColumns())
   940 		{
   941 		iTable->CopyToRowL(iRow,TPtrC8(aRPtr,aLength));
   942 		CDbBlobSpace* blobs=iTable->BlobsL();
   943 		TDbColNo col=1;
   944 		HDbColumnSet::TIteratorC iter=iTable->Def().Columns().Begin();
   945 		const HDbColumnSet::TIteratorC end=iTable->Def().Columns().End();
   946 		do
   947 			{
   948 			if (!TDbCol::IsLong(iter->Type()))
   949 				continue;
   950 			TDbBlob& blob=CONST_CAST(TDbBlob&,TDbColumnC(iRow,col).Blob());
   951 			if (blob.IsInline())
   952 				continue;
   953 			// do what has to be done...?
   954 			TUint8* data=(TUint8*)User::AllocLC(blob.Size());
   955 			blobs->ReadLC(blob.Id(),iter->Type())->ReadL(data,blob.Size());
   956 			CleanupStack::PopAndDestroy();	// stream buffer
   957 			// re-write the Blob to compress it
   958 			blobs->DeleteL(blob.Id());
   959 			blob.SetId(blobs->CreateL(iter->Type(),data,blob.Size()));
   960 			CleanupStack::PopAndDestroy();	// data
   961 			} while (++col,++iter<end);
   962 		iTable->CopyFromRow(aWPtr,iRow);
   963 		}
   964 	else
   965 		Mem::Copy(aWPtr,aRPtr,aLength);
   966 	return aWPtr+aLength;
   967 	}
   968 
   969 // Class CDbStoreCompression
   970 //
   971 // This class manages the compression for the database, applying filters as appropriate
   972 // It also defines the extrenalisation format for the huffman trees
   973 
   974 const TInt KDeflationCodes=3*(CDbStoreCompression::TEncoding::ELitLens+CDbStoreCompression::TEncoding::EDistances);
   975 
   976 inline CDbStoreCompression::CDbStoreCompression()
   977 //	:iState(EAnalysis)
   978 	{}
   979 
   980 CDbStoreCompression* CDbStoreCompression::NewL()
   981 	{
   982 	return new(ELeave) CDbStoreCompression;
   983 	}
   984 
   985 //
   986 // Build huffman codings from the freqeuncy tables
   987 //
   988 void CDbStoreCompression::EncodeL()
   989 	{
   990 	//Check the comments in CDbStoreCompression::InternalizeL().
   991 	//The implementation there is very similar to this one and is commented why the "array overrun" 
   992 	//case is impossible.
   993 	__ASSERT(iState==EAnalysis);
   994 	TUint32* p=iEncoding[0].iLitLen;
   995 	TUint32* end=p+KDeflationCodes;
   996 	do
   997 		{
   998 		Huffman::EncodingL(p,TEncoding::ELitLens);
   999 		p+=TEncoding::ELitLens;
  1000 		//coverity[overrun-local]
  1001 		Huffman::EncodingL(p,TEncoding::EDistances);
  1002 		//coverity[overrun-local]
  1003 		p+=TEncoding::EDistances;
  1004 		} while (p<end);
  1005 	iState=EEncoding;
  1006 	}
  1007 
  1008 //
  1009 // Store the encoding tables as a sequence of code lengths
  1010 // The code lengths (0-25) are themselves huffman coded, and the meta coding is stored first
  1011 //
  1012 void CDbStoreCompression::ExternalizeL(RWriteStream& aStream) const
  1013 	{
  1014 	__ASSERT(iState==EEncoding);
  1015 	const TUint32* base=iEncoding[0].iLitLen;
  1016 	const TUint32* end=base+KDeflationCodes;
  1017 	TUint32 codes[KDeflateMetaCodes];
  1018 	Mem::FillZ(codes,sizeof(codes));
  1019 	const TUint32* p=base;
  1020 	do ++codes[*p++>>KHuffMaxCodeLength]; while (p<end);
  1021 	Huffman::EncodingL(codes,KDeflateMetaCodes);
  1022 // save the meta encoding
  1023 	p=codes+KDeflateMetaCodes;
  1024 	do
  1025 		{
  1026 		TUint c0=*--p;
  1027 		TUint c1=*--p;
  1028 		c0>>=KHuffMaxCodeLength;
  1029 		c1>>=KHuffMaxCodeLength;
  1030 		aStream.WriteUint8L((c0<<4)|c1);
  1031 		} while (p>codes);
  1032 // write the encoding
  1033 	THuffEncoder encoder(aStream);
  1034 	p=base;
  1035 	do encoder.EncodeL(codes[*p++>>KHuffMaxCodeLength]); while (p<end);
  1036 	encoder.CompleteL();
  1037 	}
  1038 
  1039 //
  1040 // Internalize a previous saved encoding
  1041 //
  1042 void CDbStoreCompression::InternalizeL(RReadStream& aStream)
  1043 	{
  1044 	__ASSERT(iState!=EEncoding);
  1045 //
  1046 // read the meta encoding
  1047 	TUint32 decode[KDeflateMetaCodes];
  1048 	TUint32* p=decode+KDeflateMetaCodes;
  1049 	do
  1050 		{
  1051 		TUint8 c=aStream.ReadUint8L();
  1052 		*--p=c>>4;
  1053 		*--p=c&0xf;
  1054 		} while (p>decode);
  1055 	Huffman::Decoding(decode,KDeflateMetaCodes);
  1056 // decode the encoding
  1057 	p=iEncoding[0].iLitLen;
  1058 	TUint32* end=p+KDeflationCodes;
  1059 	TUint bits=KBitsInit;
  1060 	do
  1061 		{
  1062 		const TUint32* node=decode;
  1063 		TUint huff;
  1064 		for (;;)
  1065 			{
  1066 			huff=*node++;
  1067 			if ((bits<<=1)&KBitsEmpty)
  1068 				bits=aStream.ReadUint8L()|KBitsFull;
  1069 			if (bits&KBitsNext)
  1070 				{
  1071 				if (huff&(KHuffTerminate<<16))
  1072 					break;
  1073 				}
  1074 			else
  1075 				{
  1076 				if (huff&KHuffTerminate)
  1077 					{
  1078 					huff<<=16;
  1079 					break;
  1080 					}
  1081 				else
  1082 					node=PtrAdd(node,huff);
  1083 				}
  1084 			}
  1085 		*p++=huff>>17;
  1086 		} while (p<end);
  1087 // convert the length tables into huffman decoding trees
  1088 	//The iEncoding data member is an array of 3 elements of TEncoding type.
  1089 	//The TEncoding layout is: TUint32 iLitLen[ELitLens], TUint32 iDistance[EDistances].
  1090 	//Then the in-memory presentation of iEncoding is:
  1091 	//               ELitLens     EDistances
  1092 	//iEncoding[0]   ........     ........ 
  1093 	//iEncoding[1]   ........     ........ 
  1094 	//iEncoding[2]   ........     ........
  1095 	//
  1096 	//Bellow, in the loop:
  1097 	// p+=TEncoding::ELitLens;   - moves the pointer to the beginning of iDistance data
  1098 	// p+=TEncoding::EDistances; - moves the pointer to the beginning of iLitLen data of the next element of iEncoding.
  1099 	//The loop will process the data until "p<end", and "end" is defined as:
  1100 	// p=iEncoding[0].iLitLen;
  1101 	// TUint32* end=p+KDeflationCodes;
  1102 	//KDeflationCodes value is: 3*(CDbStoreCompression::TEncoding::ELitLens+CDbStoreCompression::TEncoding::EDistances),
  1103 	//so that is the end of the iEncoding array.
  1104 	//Conclusion: the code incrementing the "p" pointer in the loop bellow won't cause array overrun. 
  1105 	p=iEncoding[0].iLitLen;
  1106 	do
  1107 		{
  1108 		Huffman::Decoding(p,TEncoding::ELitLens);
  1109 		p+=TEncoding::ELitLens;
  1110 		//coverity[overrun-local]
  1111 		Huffman::Decoding(p,TEncoding::EDistances,KDeflateDistCodeBase);
  1112 		//coverity[overrun-local]
  1113 		p+=TEncoding::EDistances;
  1114 		} while (p<end);
  1115 	if (iState==EAnalysis)
  1116 		iState=EDecoding;
  1117 	}
  1118 
  1119 //
  1120 // Apply an inflation filter to a read stream
  1121 //
  1122 MStreamBuf* CDbStoreCompression::FilterL(MStreamBuf* aHost,TUint32,RDbStoreReadStream::TType aType)
  1123 	{
  1124 	if (iState==EDecoding || iState==EInflating)
  1125 		return HInflateBuf::NewL(aHost,iEncoding[aType]);
  1126 	return aHost;
  1127 	}
  1128 
  1129 //
  1130 // Apply a statistics or inflation filter to a write stream
  1131 //
  1132 MStreamBuf* CDbStoreCompression::FilterL(MStreamBuf* aHost,TUint32,RDbStoreWriteStream::TType aType)
  1133 	{
  1134 	TState s=iState;
  1135 	if (s==EDecoding)
  1136 		__LEAVE(KErrWrite);		// read-only database
  1137 	else if (s!=EInflating)
  1138 		{
  1139 		__ASSERT(TInt(EAnalysis)==TInt(HDeflateBuf::EAnalysis));
  1140 		__ASSERT(TInt(EEncoding)==TInt(HDeflateBuf::EDeflate));
  1141 		return HDeflateBuf::NewL(aHost,iEncoding[aType],HDeflateBuf::TMode(s));
  1142 		}
  1143 	return aHost;
  1144 	}
  1145 
  1146 // Class CDbStoreDatabase
  1147 //
  1148 // Compression related code is maintained in this source file
  1149 
  1150 //
  1151 // Iterate across all tables applying analysis or compression to them
  1152 //
  1153 void CDbStoreDatabase::CompressTablesL()
  1154 	{
  1155 	TSglQueIterC<CDbStoreDef> iter(SchemaL());
  1156 	const CDbStoreDef* def;
  1157 	while ((def=iter++)!=0)
  1158 		{
  1159 		CDbStoreTable::CCompressor* comp=new(ELeave) CDbStoreTable::CCompressor;
  1160 		CleanupStack::PushL(comp);
  1161 		comp->ProcessL(STATIC_CAST(CDbStoreTable*,TableL(*def)));
  1162 		CleanupStack::PopAndDestroy();	// comp
  1163 		}
  1164 	}
  1165 
  1166 //
  1167 // Compress or decompress the whole database
  1168 //
  1169 void CDbStoreDatabase::CompressL(TStreamId aStreamId,TZipType aZip)
  1170 	{
  1171 	__ASSERT(iStore);
  1172 	iSchemaId=aStreamId;
  1173 // read the databse header for encryption information
  1174 	RStoreReadStream strm;
  1175 	strm.OpenLC(Store(),aStreamId);
  1176 	ReadHeaderL(strm);
  1177 	CleanupStack::PopAndDestroy();	// strm
  1178 	InitPagePoolL();
  1179 //
  1180 	if (iVersion==EDbStoreCompressed)
  1181 		{
  1182 		iCompression->Inflate();
  1183 		if (aZip==EDeflate)
  1184 			__LEAVE(KErrArgument);		// already compressed
  1185 		}
  1186 	else if (aZip==EInflate)
  1187 		__LEAVE(KErrArgument);		// not compressed
  1188 	else
  1189 		{	// deflate pass #1: analyse the database
  1190 		CompressionL();				// construct the compression filter
  1191 		Transaction().DDLBeginLC();
  1192 		CompressTablesL();
  1193 		iClusterCache->FlushL();		// force through the stats buffer
  1194 		ReplaceSchemaL();				// force through the stats buffer
  1195 		CleanupStack::PopAndDestroy();	// rollback after analysis!
  1196 		iCompression->EncodeL();
  1197 		}
  1198 // now inflate or deflate the data
  1199 	Transaction().DDLBeginLC();
  1200 	CompressTablesL();
  1201 	iVersion=TUint8(aZip==EDeflate ? EDbStoreCompressed : EDbStoreVersion2);
  1202 	Transaction().DDLCommitL();
  1203 	CleanupStack::Pop();		// rollback not required
  1204 	}
  1205 
  1206 void CDbStoreDatabase::CompressL(CStreamStore* aStore,TStreamId aStreamId,TZipType aZip)
  1207 	{
  1208 	//It looks like there is a potential memory leak in the next 4 lines of code, because:
  1209 	//1) CDbStoreDatabase* self=NewLC(aStore) - new CDbStoreDatabase object is created on the heap
  1210 	//2) CDbObject* db=self->InterfaceL() - new CDbObject is created on the heap 
  1211 	//3) CleanupStack::Pop() - the CDbStoreDatabase object from (1) - out from the cleanup stack
  1212 	//4) db->PushL() - the CDbObject from (2) - in the cleanup stack
  1213 	//If one of the DDLPrepareL() or CompressL() leaves, looks like the CDbStoreDatabase object from (1) will be leaked.
  1214 	//This is not a correct guess, becuse:
  1215 	// -  CDbObject* db=self->InterfaceL() - this call creates a new CInterface object 
  1216 	//                                       on the heap. The CInterface() constructor will store the "this" pointer 
  1217 	//                                       passed from the InterfaceL() (so the "self" pointer),
  1218 	//	                                     into its "iDatabase" private data member.
  1219 	//                                       In which case, the returned CDbObject, of CInterface type, will have
  1220 	//                                       its "iDatabase" data member initialized.
  1221 	//                                       The inheritance tree is:
  1222 	//                                           CBase <-- CDbObject <-- CDbDatabase <-- CInterface
  1223 	// - db->PushL() - this call puts on the cleanup stack CDbObject::Destroy().
  1224 	//                                       The "db" object which real type is CInterface with "iDatabase" data member
  1225 	//                                       initialized with "self", is on the cleanup stack.
  1226 	// - If one of the next "L" functions leaves, then CDbObject::Destroy() will be executed.
  1227 	// - CDbObject::Destroy() will call CInterface::~CInterface() - CBase is at the root of the inheritance tree, with
  1228 	//                                       virtual ~CBase() destructor.
  1229 	// - CInterface::~CInterface() implementation calls Database().Close(), so that is CDbStoreDatabase::Close() called on 
  1230 	//                                       the "self" pointer.
  1231 	// - The CDbStoreDatabase::Close() will call "delete this" when its reference count reaches 0.
  1232 	//                                       The reference counter is increased by 1 in the InterfaceL() chain of calls.
  1233 	// -------- Conclusion ---------
  1234 	// No memory leak will occur in the next lines of code. Coverity errors - disabled.
  1235 	CDbStoreDatabase* self=NewLC(aStore);
  1236 	//coverity[alloc_fn]
  1237 	//coverity[var_assign]
  1238 	CDbObject* db=self->InterfaceL();	// a reference to the database is required
  1239 	CleanupStack::Pop();	// self
  1240 	//coverity[noescape]
  1241 	db->PushL();
  1242 	//coverity[noescape]
  1243 	self->Transaction().DDLPrepareL(*db);
  1244 	self->CompressL(aStreamId,aZip);
  1245 	CleanupStack::PopAndDestroy();	// db
  1246 	//coverity[leaked_storage]
  1247 	}
  1248 
  1249 // Class RDbStoreDatabase
  1250 
  1251 EXPORT_C void RDbStoreDatabase::CompressL(CStreamStore& aStore,TStreamId aId)
  1252 	{
  1253 	CDbStoreDatabase::CompressL(&aStore,aId,CDbStoreDatabase::EDeflate);
  1254 	}
  1255 
  1256 EXPORT_C void RDbStoreDatabase::DecompressL(CStreamStore& aStore,TStreamId aId)
  1257 	{
  1258 	CDbStoreDatabase::CompressL(&aStore,aId,CDbStoreDatabase::EInflate);
  1259 	}