os/persistentdata/persistentstorage/dbms/ustor/US_TABLE.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 //
    15 
    16 #include "US_STD.H"
    17 
    18 //#define __READBIT_CLASS
    19 //#define __WRITEBIT_CLASS
    20 
    21 #ifdef __READBIT_CLASS
    22 class TReadBitSequence
    23 	{
    24 public:
    25 	inline TReadBitSequence();
    26 	inline TBool HasBits() const;
    27 	TUint Read(const TUint8*& aPtr);
    28 private:
    29 	TUint iBits;
    30 	};
    31 inline TReadBitSequence::TReadBitSequence()
    32 	:iBits(0) {}
    33 inline TBool TReadBitSequence::HasBits() const
    34 	{return (iBits&0x2000000);}
    35 TUint TReadBitSequence::Read(const TUint8*& aPtr)
    36 	{
    37 	iBits>>=1;
    38 	if ((iBits&0x1000000)==0)
    39 		iBits=*aPtr++ | 0xff000000u;
    40 	return iBits&1;
    41 	}
    42 #define READBITINIT(This) TReadBitSequence This
    43 #define READHASBITS(This) This.HasBits()
    44 #define READBIT(This,ptr) This.Read(ptr)
    45 #define SKIPBIT(This,ptr) READBIT(This,ptr)
    46 #else
    47 #define READBITINIT(This) TUint This##_bits=0
    48 #define READHASBITS(This) (This##_bits&0x2000000)
    49 #define SKIPBIT(This,ptr) (This##_bits>>=1,(This##_bits&0x1000000 ? void(0) : void(This##_bits=*ptr++|0xff000000u)))
    50 #define READBIT(This,ptr) (SKIPBIT(This,ptr),This##_bits&1)
    51 #endif
    52 
    53 #ifdef __WRITEBIT_CLASS
    54 class TWriteBitSequence
    55 	{
    56 public:
    57 	inline TWriteBitSequence();
    58 	TUint8* Write(TUint8* aPtr,TUint aBit);
    59 	void Flush();
    60 private:
    61 	TUint iBits;
    62 	TUint8* iPtr;
    63 	};
    64 inline TWriteBitSequence::TWriteBitSequence()
    65 	:iBits(0),iPtr(0) {}
    66 TUint8* TWriteBitSequence::Write(TUint8* aPtr,TUint aBit)
    67 	{
    68 	TUint bits=iBits>>1;
    69 	if ((bits&0x1000000)==0)
    70 		{
    71 		if (iPtr)
    72 			*iPtr=TUint8(bits);
    73 		iPtr=aPtr++;
    74 		bits=0xff000000;
    75 		}
    76 	if (aBit)
    77 		bits|=0x100;
    78 	iBits=bits;
    79 	return aPtr;
    80 	}
    81 void TWriteBitSequence::Flush()
    82 	{
    83 	TUint8* ptr=iPtr;
    84 	if (ptr)
    85 		{
    86 		TUint bits=iBits;
    87 		do bits>>=1; while (bits&0x1000000);
    88 		*ptr=TUint8(bits);
    89 		}
    90 	}
    91 #define WRITEBITINIT(This) TWriteBitSequence This
    92 #define WRITEBIT(This,ptr,bit) ptr=This.Write(ptr,bit)
    93 #define WRITEZEROBIT(This,ptr) ptr=This.Write(ptr,0)
    94 #define FLUSHBITS(This) This.Flush()
    95 #else
    96 #define WRITEBITINIT(This) TUint32 This##_bits=0; TUint8* This##_ptr=0
    97 #define NEXTBIT(This,ptr) This##_bits>>=1;if ((This##_bits&0x1000000)==0){if (This##_ptr) *This##_ptr=TUint8(This##_bits);This##_ptr=ptr++;This##_bits=0xff000000;}
    98 #define WRITEZEROBIT(This,ptr) {NEXTBIT(This,ptr)}
    99 #define WRITEBIT(This,ptr,bit) {NEXTBIT(This,ptr) if (bit) This##_bits|=0x100;}
   100 #define FLUSHBITS(This) {if (This##_ptr){do This##_bits>>=1; while (This##_bits&0x1000000);*This##_ptr=TUint8(This##_bits);}}
   101 #endif
   102 
   103 LOCAL_C const TUint8* Read32(const TUint8* aPtr,TUint32* aDest)
   104 // fast read for non-aligned little-endian 32-bit data
   105 	{
   106 	TUint32 x=*aPtr++;
   107 	x|=*aPtr++<<8;
   108 	x|=*aPtr++<<16;
   109 	x|=*aPtr++<<24;
   110 	*aDest=x;
   111 	return aPtr;
   112 	}
   113 
   114 LOCAL_C TUint8* Write32(TUint8* aPtr,TUint32 aVal)
   115 // fast transfer for non-aligned little-endian 32-bit data
   116 	{
   117 	*aPtr++=TUint8(aVal);
   118 	*aPtr++=TUint8(aVal>>8);
   119 	*aPtr++=TUint8(aVal>>16);
   120 	*aPtr++=TUint8(aVal>>24);
   121 	return aPtr;
   122 	}
   123 
   124 inline const TUint8* Read16(const TUint8* aPtr,TUint32* aDest)
   125 // Read unsigned 16-bit value into aDest storage
   126 	{
   127 	TUint x=*aPtr++;
   128 	x|=*aPtr++<<8;
   129 	*aDest=TUint16(x);
   130 	return aPtr;
   131 	}
   132 
   133 inline const TUint8* Read16s(const TUint8* aPtr,TUint32* aDest)
   134 // Read signed 16-bit value into aDest storage
   135 	{
   136 	TInt x=*aPtr++<<16;
   137 	x|=*aPtr++<<24;
   138 	*aDest=x>>16;
   139 	return aPtr;
   140 	}
   141 
   142 inline TUint8* Write16(TUint8* aPtr,TUint aVal)
   143 // Write little-endian rep of the low 16 bits of aVal
   144 	{
   145 	*aPtr++=TUint8(aVal);
   146 	*aPtr++=TUint8(aVal>>8);
   147 	return aPtr;
   148 	}
   149 
   150 LOCAL_C TUint8* WriteCardinality(TUint8* aPtr,TUint aVal)
   151 // compress cardinality into the data stream
   152 	{
   153 	__ASSERT(aVal<(1u<<30));
   154 	if ((aVal>>7)==0)
   155 		{
   156 		*aPtr++=TUint8(aVal<<1);
   157 		return aPtr;
   158 		}
   159 	aVal=(aVal<<2)|1;
   160 	if ((aVal>>16)==0)
   161 		{
   162 		*aPtr++=TUint8(aVal);
   163 		*aPtr++=TUint8(aVal>>8);
   164 		return aPtr;
   165 		}
   166 	return Write32(aPtr,aVal|2);
   167 	}
   168 
   169 LOCAL_C TUint ReadCardinality(const TUint8* aPtr)
   170 // extract cardinality from the data stream
   171 	{
   172 	TUint x=aPtr[0];
   173 	if ((x&1)==0)
   174 		return x>>1;
   175 	x|=aPtr[1]<<8;
   176 	if (x&2)
   177 		{
   178 		x|=aPtr[2]<<16;
   179 		x|=aPtr[3]<<24;
   180 		}
   181 	return x>>2;
   182 	}
   183 
   184 LOCAL_C const TUint8* ReadCardinality(const TUint8* aPtr,TInt& aVal)
   185 // extract cardinality from the data stream
   186 	{
   187 	TUint x=*aPtr++;
   188 	if ((x&1)==0)
   189 		x>>=1;
   190 	else
   191 		{
   192 		x|=*aPtr++<<8;
   193 		if (x&2)
   194 			{
   195 			x|=*aPtr++<<16;
   196 			x|=*aPtr++<<24;
   197 			}
   198 		x>>=2;
   199 		}
   200 	aVal=x;
   201 	return aPtr;
   202 	}
   203 
   204 LOCAL_C TInt SizeOfCardinality(TUint aVal)
   205 // report the externalized size of the Compressed value in bytes
   206 	{
   207 	if ((aVal>>7)==0)
   208 		return 1;
   209 	return (aVal>>14)==0 ? 2 : 4;
   210 	}
   211 
   212 LOCAL_C TUint8* WriteBlobId(TUint8* aPtr,TDbBlobId aId)
   213 	{
   214 	return WriteCardinality(aPtr,(aId>>24)|(aId<<8>>4));
   215 	}
   216 
   217 LOCAL_C const TUint8* ReadBlobId(const TUint8* aPtr,TDbBlobId& aId)
   218 	{
   219 	aPtr=ReadCardinality(aPtr,*reinterpret_cast<TInt*>(&aId));
   220 	aId=(aId>>4)|(aId<<28>>4);
   221 	return aPtr;
   222 	}
   223 
   224 LOCAL_C TInt SizeOfBlobId(TDbBlobId aId)
   225 	{
   226 	return SizeOfCardinality((aId>>24)|(aId<<8>>4));
   227 	}
   228 
   229 // Class CDbStoreTable
   230 
   231 CDbStoreTable::CDbStoreTable(CDbStoreDatabase& aDatabase,const CDbTableDef& aDef)
   232 	: CDbTable(aDatabase,aDef)
   233 	{}
   234 
   235 CDbRecordSpace* CDbStoreTable::RecordSpaceL()
   236 //
   237 // open records handler
   238 //
   239 	{
   240 	return CDbStoreRecords::NewL(Database().ClusterCacheL(),Def());
   241 	}
   242 
   243 CDbBlobSpace* CDbStoreTable::BlobSpaceL()
   244 //
   245 // Open a blobs accessor for the table
   246 //
   247 	{
   248 	return new(ELeave) CDbStoreBlobs(Database(),Def().InlineLimit());
   249 	}
   250 
   251 CDbRecordIndex* CDbStoreTable::RecordIndexL(const CDbTableIndexDef& anIndex)
   252 //
   253 // Open an index
   254 //
   255 	{
   256 	return CDbStoreIndex::NewL(Database(),(const CDbStoreIndexDef&)anIndex,Def());
   257 	}
   258 
   259 static TUint8* CompressUnicode(TUint8* aRec,const TAny* aData,TInt aSize)
   260 //
   261 // initially assume the compressed data requires 1 byte for the length data
   262 // Copy it if more is necessary. This avoids having to run the compressor twice
   263 //
   264 	{
   265 	TMemoryUnicodeSource source(reinterpret_cast<const TUint16*>(aData));
   266 	TUnicodeCompressor compressor;
   267 	TInt output;
   268 	compressor.CompressL(aRec+1,source,KMaxTInt,aSize>>1,&output);
   269 	TInt lenSize=SizeOfCardinality(output);
   270 	if (lenSize!=1)
   271 		Mem::Copy(aRec+lenSize,aRec+1,output);	// needs more space for length
   272 	return WriteCardinality(aRec,output)+output;
   273 	}
   274 
   275 static TInt SizeOfCompressedUnicode(const TAny* aData,TInt aSize)
   276 //
   277 // bytes required to store the unicode data
   278 //
   279 	{
   280 	TMemoryUnicodeSource source(reinterpret_cast<const TUint16*>(aData));
   281 	TInt size=TUnicodeCompressor::CompressedSizeL(source,aSize>>1);
   282 	return SizeOfCardinality(size)+size;
   283 	}
   284 
   285 static const TUint8* ExpandUnicodeL(const TUint8* aRec,TAny* aTarget,const TAny* aLimit,TInt& aChars)
   286 //
   287 // Read compressed unicode from the record
   288 //
   289 	{
   290 	TInt bytes;
   291 	aRec=ReadCardinality(aRec,bytes);
   292 	TMemoryUnicodeSink unicode(reinterpret_cast<TUint16*>(aTarget));
   293 	TUnicodeExpander expander;
   294 	TInt used;
   295 	expander.ExpandL(unicode,aRec,(TUint16*)aLimit-(TUint16*)aTarget,bytes,&aChars,&used);
   296 	return bytes==used ? aRec+bytes : 0;	// signal corruption in data, could not fit text in space
   297 	}
   298 
   299 static TInt SizeOfExpandedUnicodeL(const TUint8* aData,TInt aSize)
   300 //
   301 // bytes required to store the unicode data
   302 //
   303 	{
   304 	return TUnicodeExpander::ExpandedSizeL(aData,aSize)<<1;
   305 	}
   306 
   307 TInt CDbStoreTable::RecordLength(const RDbRow& aRow)
   308 //
   309 // Calculate the size of a record
   310 //
   311 	{
   312 	TInt bits=SizeOfCardinality(OptimizedRowLength(aRow))<<3;		// record buffer size
   313 	HDbColumnSet::TIteratorC col=Def().Columns().Begin();
   314 	const TDbCell* const last=aRow.Last();
   315 	for (const TDbCell* column=aRow.First();column<last;++col,column=column->Next())
   316 		{
   317 		__ASSERT(col<Def().Columns().End());	// columns off end
   318 		TInt size=column->Length();
   319 		if ((col->iAttributes&TDbCol::ENotNull)==0)
   320 			{	//nullable
   321 			++bits;
   322 			if (size==0)
   323 				continue;			// no data
   324 			}
   325 		__ASSERT(size>0);	// must be non-null to reach here
   326 		TDbColType type=col->Type();
   327 		__ASSERT(type>=EDbColBit&&type<=EDbColLongBinary);
   328 		if (type==EDbColBit)
   329 			++bits;
   330 		else if (type<=EDbColDateTime)
   331 			bits+=TRecordSize::FixedFieldSize(type)<<3;
   332 		else if (type==EDbColText16)
   333 			bits+=SizeOfCompressedUnicode(column->Data(),size)<<3;
   334 		else if (type<=EDbColBinary)
   335 			bits+=8+(size<<3);
   336 		else
   337 			{
   338 			__ASSERT(type<=EDbColLongBinary);
   339 			const TDbBlob& blob=*(const TDbBlob*)column->Data();
   340 			if (!blob.IsInline())
   341 				bits+=1+((SizeOfBlobId(blob.Id())+SizeOfCardinality(blob.Size()))<<3);
   342 			else if (type!=EDbColLongText16)
   343 				bits+=9+(blob.Size()<<3);
   344 			else
   345 				bits+=1+(SizeOfCompressedUnicode(blob.Data(),blob.Size())<<3);
   346 			}
   347 		}
   348 	__ASSERT(bits<=(KDbStoreMaxRecordLength<<3));
   349 	return (bits+7)>>3;
   350 	}
   351 
   352 TInt CDbStoreTable::OptimizedRowLength(const RDbRow& aRow)
   353 //
   354 // Calculate the minimal row buffer size (in words) to store the row data
   355 //
   356 	{
   357 	HDbColumnSet::TIteratorC col=Def().Columns().Begin();
   358 	const TDbCell* const last=aRow.Last();
   359 	TInt cellHead=0;
   360 	TInt words=0;
   361 	for (const TDbCell* column=aRow.First();column<last;++col,column=column->Next())
   362 		{
   363 		++cellHead;
   364 		__ASSERT(col<Def().Columns().End());	// columns off end
   365 		TInt size=column->Length();
   366 		if (size==0)
   367 			continue;
   368 		words+=cellHead;
   369 		cellHead=0;
   370 		TDbColType type=col->Type();
   371 		__ASSERT(type>=EDbColBit&&type<=EDbColLongBinary);
   372 		if (type<=EDbColDateTime)
   373 			__ASSERT(size==(size>>2<<2));
   374 		else if (type<=EDbColBinary)
   375 			;
   376 		else
   377 			size=((const TDbBlob*)column->Data())->CellSize();
   378 		words+=(size+3)>>2;
   379 		}
   380 	return words;
   381 	}
   382 
   383 void CDbStoreTable::CopyFromRow(TUint8* aRecord,const RDbRow& aRow)
   384 //
   385 // translate the row buffer into its persistent format in the record buffer
   386 //
   387 	{
   388 	aRecord=WriteCardinality(aRecord,OptimizedRowLength(aRow));	// internal size
   389 //
   390 	WRITEBITINIT(bits);
   391 	HDbColumnSet::TIteratorC iter=Def().Columns().Begin();
   392 	const TDbCell* const last=aRow.Last();
   393 	for (const TDbCell* column=aRow.First();column<last;++iter,column=column->Next())
   394 		{
   395 		__ASSERT(iter<Def().Columns().End());	// columns off end
   396 		TInt size=column->Length();
   397 		if ((iter->iAttributes&TDbCol::ENotNull)==0)
   398 			{	// nullable
   399 			WRITEBIT(bits,aRecord,size!=0);
   400 			if (size==0)
   401 				continue;			// no data
   402 			}
   403 		__ASSERT(size>0);	// must be non-null to reach here
   404 		const TUint32* data=(const TUint32*)column->Data();
   405 		TDbColType type=iter->Type();
   406 		switch (type)
   407 			{
   408 		default:
   409 			__ASSERT(0);
   410 		case EDbColBit:
   411 			WRITEBIT(bits,aRecord,*data);
   412 			break;
   413 		case EDbColInt8:
   414 		case EDbColUint8:
   415 			*aRecord++=TUint8(*data);
   416 			break;
   417 		case EDbColInt16:
   418 		case EDbColUint16:
   419 			aRecord=Write16(aRecord,*data);
   420 			break;
   421 #if defined(__DOUBLE_WORDS_SWAPPED__)
   422 		case EDbColReal64:
   423 			aRecord=Write32(aRecord,data[1]);	// write low word out first
   424 			// drop through to write high word next
   425 #endif
   426 		case EDbColInt32:
   427 		case EDbColUint32:
   428 		case EDbColReal32:
   429 			aRecord=Write32(aRecord,*data);
   430 			break;
   431 #if !defined(__DOUBLE_WORDS_SWAPPED__)
   432 		case EDbColReal64:
   433 #endif
   434 		case EDbColInt64:
   435 		case EDbColDateTime:
   436 			aRecord=Write32(aRecord,data[0]);
   437 			aRecord=Write32(aRecord,data[1]);
   438 			break;
   439 		case EDbColText16:
   440 			aRecord=CompressUnicode(aRecord,data,size);
   441 			break;
   442 		case EDbColText8:
   443 		case EDbColBinary:
   444 			*aRecord++=TUint8(size);
   445 			aRecord=Mem::Copy(aRecord,data,size);
   446 			break;
   447 		case EDbColLongText8:
   448 		case EDbColLongText16:
   449 		case EDbColLongBinary:
   450 			{
   451 			const TDbBlob& blob=*reinterpret_cast<const TDbBlob*>(data);
   452 			size=blob.Size();
   453 			WRITEBIT(bits,aRecord,blob.IsInline());
   454 			if (blob.IsInline())
   455 				{
   456 				if (type==EDbColLongText16)
   457 					aRecord=CompressUnicode(aRecord,blob.Data(),size);
   458 				else
   459 					{
   460 					*aRecord++=TUint8(size);
   461 					aRecord=Mem::Copy(aRecord,blob.Data(),size);
   462 					}
   463 				}
   464 			else
   465 				{
   466 				aRecord=WriteBlobId(aRecord,blob.Id());
   467 				aRecord=WriteCardinality(aRecord,size);
   468 				}
   469 			}
   470 			break;
   471 			}
   472 		}
   473 	FLUSHBITS(bits);
   474 	}
   475 
   476 const TUint8* CDbStoreTable::CopyToRowL(TDbCell* aCell,TInt aSize,const TUint8* aRec)
   477 //
   478 // translate persistent record into the row buffer
   479 //
   480 	{
   481 	__ASSERT(aSize>0);
   482 //
   483 	const TDbCell* const end=PtrAdd(aCell,aSize);
   484 	READBITINIT(bits);
   485 	HDbColumnSet::TIteratorC col=Def().Columns().Begin();
   486 	HDbColumnSet::TIteratorC const cend=Def().Columns().End();
   487 	for (;;)
   488 		{
   489 		TInt size=0; //null data
   490 		if (col->iAttributes&TDbCol::ENotNull || READBIT(bits,aRec))	// have data
   491 			{
   492 			if (TInt(end)-TInt(aCell)<=(TInt)sizeof(TUint32))
   493 				return 0;
   494 			size=sizeof(TUint32);	// for most types
   495 			TDbColType type=col->Type();
   496 			switch (type)
   497 				{
   498 			default:
   499 				__ASSERT(0);
   500 			case EDbColBit:
   501 				*(TUint32*)aCell->Data()=READBIT(bits,aRec);
   502 				break;
   503 			case EDbColInt8:
   504 				*(TInt32*)aCell->Data()=*(const TInt8*)aRec;
   505 				aRec+=sizeof(TInt8);
   506 				break;
   507 			case EDbColUint8:
   508 				*(TUint32*)aCell->Data()=*aRec;
   509 				aRec+=sizeof(TUint8);
   510 				break;
   511 			case EDbColInt16:
   512 				aRec=Read16s(aRec,(TUint32*)aCell->Data());
   513 				break;
   514 			case EDbColUint16:
   515 				aRec=Read16(aRec,(TUint32*)aCell->Data());
   516 				break;
   517 	#if defined(__DOUBLE_WORDS_SWAPPED__)
   518 			case EDbColReal64:
   519 				if (TInt(end)-TInt(aCell)<=(TInt)sizeof(TReal64))
   520 					return 0;
   521 				size=sizeof(TReal64);
   522 				aRec=Read32(aRec,(TUint32*)aCell->Data()+1);	// transfer low word first, to high address
   523 				// drop through to transfer high word to low address!
   524 	#endif
   525 			case EDbColInt32:
   526 			case EDbColUint32:
   527 			case EDbColReal32:
   528 				aRec=Read32(aRec,(TUint32*)aCell->Data());
   529 				break;
   530 	#if !defined(__DOUBLE_WORDS_SWAPPED__)
   531 			case EDbColReal64:
   532 	#endif
   533 			case EDbColInt64:
   534 			case EDbColDateTime:
   535 				if (TInt(end)-TInt(aCell)<=(TInt)sizeof(TInt64))
   536 					return 0;
   537 				size=sizeof(TInt64);
   538 				aRec=Read32(aRec,(TUint32*)aCell->Data());
   539 				aRec=Read32(aRec,(TUint32*)aCell->Data()+1);
   540 				break;
   541 			case EDbColText16:
   542 				{
   543 				TInt len;
   544 				aRec=ExpandUnicodeL(aRec,aCell->Data(),end,len);
   545 				if (!aRec)
   546 					return 0;
   547 				size=len<<1;
   548 				}
   549 				break;
   550 			case EDbColText8:
   551 			case EDbColBinary:
   552 				size=*aRec++;
   553 				if (TInt(end)-TInt(aCell)<TInt(size+sizeof(TUint32)))
   554 					return 0;
   555 				Mem::Copy(aCell->Data(),aRec,size);
   556 				aRec+=size;
   557 				break;
   558 			case EDbColLongText8:
   559 			case EDbColLongText16:
   560 			case EDbColLongBinary:
   561 				if (READBIT(bits,aRec)==0)
   562 					{	// out of line Long column
   563 					if (TInt(end)-TInt(aCell)<=TDbBlob::RefSize())
   564 						return 0;
   565 					TDbBlobId id;
   566 					aRec=ReadBlobId(aRec,id);
   567 					TInt sz;
   568 					aRec=ReadCardinality(aRec,sz);
   569 					new(aCell->Data()) TDbBlob(id,sz);
   570 					size=TDbBlob::RefSize();
   571 					}
   572 				else if (type!=EDbColLongText16)
   573 					{	// inline
   574 					size=*aRec++;
   575 					if (TInt(end)-TInt(aCell)<TInt(TDbBlob::InlineSize(size)+sizeof(TUint32)))
   576 						return 0;
   577 					new(aCell->Data()) TDbBlob(aRec,size);
   578 					aRec+=size;
   579 					size=TDbBlob::InlineSize(size);
   580 					}
   581 				else
   582 					{
   583 					TDbBlob* blob=new(aCell->Data()) TDbBlob;
   584 					TInt len;
   585 					aRec=ExpandUnicodeL(aRec,blob->InlineBuffer(),end,len);
   586 					if (!aRec)
   587 						return 0;
   588 					size=len<<1;
   589 					blob->SetSize(size);
   590 					size=TDbBlob::InlineSize(size);
   591 					}
   592 				break;
   593 				}
   594 			}
   595 		aCell->SetLength(size);
   596 		aCell=aCell->Next();
   597 		if (aCell==end)
   598 			return aRec;
   599 		if (++col==cend)
   600 			return 0;
   601 		}
   602 	}
   603 
   604 void CDbStoreTable::CopyToRowL(RDbRow& aRow,const TDesC8& aRecord)
   605 //
   606 // translate persistent record into the row buffer
   607 //
   608 	{
   609 	const TUint8* rec=aRecord.Ptr();
   610 	const TUint8* end=rec+aRecord.Length();
   611 	TInt size;
   612 	rec=ReadCardinality(rec,size);
   613 	size<<=2;
   614 	
   615 	//If such huge allocation is requested(KMaxTInt/2), it is highly likely the file is corrupt
   616 	if((size < 0) || (size >= KMaxTInt/2))
   617 		{
   618 		aRow.SetSize(0);
   619 		__LEAVE(KErrCorrupt);
   620 		}
   621 	
   622 	if (size)
   623 		{
   624 		aRow.GrowL(size);
   625 		rec=CopyToRowL(aRow.First(),size,rec);
   626 		}
   627 	if (rec)
   628 		{
   629 		do
   630 			{
   631 			if (rec==end)
   632 				{
   633 	aRow.SetSize(size);
   634 				return;
   635 				}
   636 			} while (*rec++==0);
   637 		}
   638 	aRow.SetSize(0);
   639 	__LEAVE(KErrCorrupt);
   640 	}
   641 
   642 TUint8* CDbStoreTable::AlterRecordL(TUint8* aWPtr,const TUint8* aRPtr,TInt aLength,TInt aInlineLimit)
   643 //
   644 // Rewrite the record in the buffer by removing the data from the delete-list
   645 // any changes are recorded in the iAltered member
   646 //
   647 	{
   648 	// initially assume that the length count will be the same size after alteration
   649 	TInt lenSize=SizeOfCardinality(ReadCardinality(aRPtr));
   650 	const TUint8* const rEnd=aRPtr+aLength;
   651 	aRPtr+=lenSize;
   652 	TUint8* wptr=aWPtr+lenSize;
   653 	TInt wRowSize=0;
   654 	TInt headers=0;
   655 //
   656 	READBITINIT(rbits);
   657 	WRITEBITINIT(wbits);
   658 	const HDbColumnSet::TIteratorC cEnd=Def().Columns().End();
   659 	HDbColumnSet::TIteratorC col=Def().Columns().Begin();
   660 	do
   661 		{
   662 		if (aRPtr==rEnd && !READHASBITS(rbits))
   663 			break;	// no more data
   664 		TUint flg=col->iFlags;
   665 		if ((flg&TDbColumnDef::EDropped)==0)
   666 			++headers;
   667 		if ((col->iAttributes&TDbCol::ENotNull)==0)
   668 			{	// nullable
   669 			TUint notnull=READBIT(rbits,aRPtr);
   670 			if ((flg&TDbColumnDef::EDropped)==0)
   671 				WRITEBIT(wbits,wptr,notnull);
   672 			if (notnull==0)	// null data
   673 				continue;
   674 			}
   675 		wRowSize+=headers;
   676 		headers=0;
   677 		TInt size;
   678 		TDbColType type=col->Type();
   679 		switch (type)
   680 			{
   681 		default:
   682 			__ASSERT(0);
   683 		case EDbColBit:
   684 			size=READBIT(rbits,aRPtr);
   685 			if ((flg&TDbColumnDef::EDropped)==0)
   686 				{
   687 				WRITEBIT(wbits,wptr,size);
   688 				++wRowSize;
   689 				}
   690 			continue;
   691 		case EDbColInt8:
   692 		case EDbColUint8:
   693 		case EDbColInt16:
   694 		case EDbColUint16:
   695 		case EDbColInt32:
   696 		case EDbColUint32:
   697 		case EDbColReal32:
   698 		case EDbColReal64:
   699 		case EDbColInt64:
   700 		case EDbColDateTime:
   701 			size=TRecordSize::FixedFieldSize(type);
   702 			if ((flg&TDbColumnDef::EDropped)==0)
   703 				{
   704 				wptr=Mem::Copy(wptr,aRPtr,size);
   705 				wRowSize+=(size+3)>>2;	// # words
   706 				}
   707 			break;
   708 		case EDbColText8:
   709 		case EDbColBinary:
   710 			size=*aRPtr++;
   711 			if ((flg&(TDbColumnDef::EChangedType|TDbColumnDef::EDropped))==0)
   712 				{
   713 				wptr=Mem::Copy(wptr,aRPtr-1,size+1);	// no change, copy the column
   714 				wRowSize+=(size+3)>>2;	// # words
   715 				}
   716 			else if (flg&TDbColumnDef::EChangedType)
   717 				goto alterBlob8;	// type change, into a LongColumn
   718 			else
   719 				__ASSERT(flg&TDbColumnDef::EDropped);	// drop the column
   720 			break;
   721 		case EDbColText16:
   722 			{
   723 			TInt sz;
   724 			aRPtr=ReadCardinality(aRPtr,sz);
   725 			size=sz;
   726 			if ((flg&(TDbColumnDef::EChangedType|TDbColumnDef::EDropped))==0)
   727 				{
   728 				wptr=WriteCardinality(wptr,size);
   729 				wptr=Mem::Copy(wptr,aRPtr,size);	// no change, copy the column
   730 				wRowSize+=(SizeOfExpandedUnicodeL(aRPtr,size)+3)>>2;
   731 				}
   732 			else if (flg&TDbColumnDef::EChangedType)
   733 				goto alterBlob16;	// type change, into a LongColumn
   734 			else
   735 				__ASSERT(flg&TDbColumnDef::EDropped);	// drop the column
   736 			}
   737 			break;
   738 		case EDbColLongText8:
   739 		case EDbColLongBinary:
   740 		case EDbColLongText16:
   741 			if (!READBIT(rbits,aRPtr))
   742 				{	// out-of-line
   743 				TDbBlobId id;
   744 				aRPtr=ReadBlobId(aRPtr,id);
   745 				TInt sz;
   746 				aRPtr=ReadCardinality(aRPtr,sz);
   747 				if (flg&TDbColumnDef::EDropped)
   748 					BlobsL()->DeleteL(id);	// delete the stream
   749 				else
   750 					{
   751 					WRITEZEROBIT(wbits,wptr);			// out-of-line
   752 					wptr=WriteBlobId(wptr,id);
   753 					wptr=WriteCardinality(wptr,sz);
   754 					wRowSize+=TDbBlob::RefSize()>>2;
   755 					}
   756 				size=0;
   757 				}
   758 			else if (type!=EDbColLongText16)
   759 				{	// currently inline
   760 				size=*aRPtr++;
   761 				if (flg&TDbColumnDef::EDropped)
   762 					break;
   763 				// write long-column data, check inline status
   764 alterBlob8:		WRITEBIT(wbits,wptr,size<=aInlineLimit);
   765 				if (size<=aInlineLimit)
   766 					{	// inlined
   767 					*wptr++=TUint8(size);				// blob size
   768 					wptr=Mem::Copy(wptr,aRPtr,size);	// blob data
   769 					wRowSize+=(TDbBlob::InlineSize(size)+3)>>2;
   770 					}
   771 				else
   772 					{
   773 					TDbBlobId id=BlobsL()->CreateL(type,aRPtr,size);
   774 					wptr=WriteBlobId(wptr,id);
   775 					wptr=WriteCardinality(wptr,size);
   776 					wRowSize+=TDbBlob::RefSize()>>2;
   777 					}
   778 				}
   779 			else
   780 				{	// currently inline
   781 				TInt sz;
   782 				aRPtr=ReadCardinality(aRPtr,sz);
   783 				size=sz;
   784 				if (flg&TDbColumnDef::EDropped)
   785 					break;
   786 				// write long-column data, check inline status
   787 alterBlob16:	TInt len=SizeOfExpandedUnicodeL(aRPtr,size);
   788 				WRITEBIT(wbits,wptr,len<=aInlineLimit);
   789 				if (len<=aInlineLimit)
   790 					{	// inlined
   791 					wptr=WriteCardinality(wptr,size);
   792 					wptr=Mem::Copy(wptr,aRPtr,size);	// blob data
   793 					wRowSize+=(TDbBlob::InlineSize(len)+3)>>2;
   794 					}
   795 				else
   796 					{
   797 					TDbBlobId id=BlobsL()->CreateL(EDbColLongText8,aRPtr,size);	// no unicode compressor!
   798 					wptr=WriteBlobId(wptr,id);
   799 					wptr=WriteCardinality(wptr,len);
   800 					wRowSize+=TDbBlob::RefSize()>>2;
   801 					}
   802 				}
   803 			break;
   804 			}
   805 		aRPtr+=size;
   806 		} while (++col<cEnd);
   807 	FLUSHBITS(wbits);
   808 	TInt lsz=SizeOfCardinality(wRowSize);
   809 	if (lenSize!=lsz)
   810 		wptr=Mem::Copy(aWPtr+lsz,aWPtr+lenSize,wptr-(aWPtr+lenSize));
   811 	WriteCardinality(aWPtr,wRowSize);
   812 	return wptr;
   813 	}
   814 
   815 TInt CDbStoreTable::IndexSpanL(const CDbTableIndexDef& aIndex,TUint aInclusion,const TDbLookupKey* aLower,const TDbLookupKey* aUpper)
   816 //
   817 // Guess the size of the index set contained in the given restriction
   818 // First check the cached values in the defintion, and only load the index if necessary
   819 //
   820 	{
   821 	const TDbStoreIndexStats& stats=static_cast<const CDbStoreIndexDef&>(aIndex).iStats;
   822 	if (!stats.IsValid())
   823 		IndexL(aIndex);				// ensure that the index is loaded
   824 	return stats.Span(aInclusion,aLower,aUpper,TTextOps::Ops(aIndex.Key().Comparison()));
   825 	}
   826 
   827 // Class CDbStoreTable::CDiscarder
   828 
   829 CDbStoreTable::CDiscarder::CDiscarder()
   830 	{}
   831 
   832 CDbStoreTable::CDiscarder::~CDiscarder()
   833 	{
   834 	if (iTable)
   835 		iTable->Close();
   836 	iRow.Close();
   837 	}
   838 
   839 TInt CDbStoreTable::CDiscarder::OpenL(CDbStoreTable* aTable)
   840 	{
   841 	__ASSERT(!iTable);
   842 	iTable=aTable;
   843 	iRecords=&aTable->StoreRecordsL();
   844 	iCluster=iRecords->Head();
   845 	return iRecords->Count()+1;
   846 	}
   847 
   848 TInt CDbStoreTable::CDiscarder::StepL(TInt aStep)
   849 	{
   850 	__ASSERT(iTable);
   851 	TInt limit=iTable->Def().Columns().HasLongColumns() ? EBlobDiscardClusters : EDiscardClusters;
   852 	for (TInt inc=0;inc<limit;++inc)
   853 		{
   854 		if (iCluster==KNullClusterId)
   855 			{
   856 			iRecords->DestroyL();
   857 			return 0;
   858 			}
   859 		if (limit==EBlobDiscardClusters)
   860 			iRecords->AlterL(iCluster,*this);
   861 		aStep-=iRecords->DiscardL(iCluster);
   862 		}
   863 	return Max(aStep,1);
   864 	}
   865 
   866 TUint8* CDbStoreTable::CDiscarder::AlterRecordL(TUint8* aWPtr,const TUint8* aRPtr,TInt aLength)
   867 //
   868 // Scan for and discard all known BLOBs
   869 //
   870 	{
   871 	iTable->CopyToRowL(iRow,TPtrC8(aRPtr,aLength));
   872 	iTable->DiscardBlobsL(iRow);
   873 	return aWPtr;
   874 	}
   875 
   876 // class CDbStoreTable::CAlter
   877 
   878 CDbStoreTable::CAlter::CAlter()
   879 	{}
   880 
   881 CDbStoreTable::CAlter::~CAlter()
   882 	{
   883 	if (iTable)
   884 		iTable->Close();
   885 	}
   886 
   887 void CDbStoreTable::CAlter::OpenL(CDbStoreTable* aTable,const HDbColumnSet& aNewSet)
   888 //
   889 // Prepare for alteration of the table data
   890 //
   891 	{
   892 	__ASSERT(!iTable);
   893 	iTable=aTable;
   894 	iInlineLimit=TRecordSize::InlineLimit(aNewSet);
   895 	iRecords=&iTable->StoreRecordsL();
   896 	iCluster=iRecords->Head();
   897 //
   898 // Calculate the maximum possible expansion, based on the changes to the column set
   899 // Currently the only alloed change which affects this is Text->LongText type changes
   900 // these all add a single bit
   901 //
   902 	TInt expand=0;
   903 	const HDbColumnSet& columns=iTable->Def().Columns();
   904 	const HDbColumnSet::TIteratorC end=columns.End();
   905 	HDbColumnSet::TIteratorC col=columns.Begin();
   906 	do
   907 		{
   908 		if (col->iFlags&TDbColumnDef::EChangedType)
   909 			{
   910 			__ASSERT(col->iType>=EDbColText8&&col->iType<=EDbColBinary);
   911 			++expand;
   912 			}
   913 		} while (++col<end);
   914 	iExpansion=(expand+7)>>3;
   915 	}
   916 
   917 TInt CDbStoreTable::CAlter::StepL(TInt aStep)
   918 //
   919 // Do some steps to alter the table data
   920 //
   921 	{
   922 	__ASSERT(iTable);
   923 	iStep=aStep;
   924 	iCluster=iRecords->AlterL(iCluster,*this);
   925 	return iCluster==KNullClusterId ? 0 : Max(iStep,1);
   926 	}
   927 
   928 TUint8* CDbStoreTable::CAlter::AlterRecordL(TUint8* aWPtr,const TUint8* aRPtr,TInt aLength)
   929 //
   930 // providing for CClusterCache::MAlter
   931 // re-write the record
   932 //
   933 	{
   934 	--iStep;
   935 	return iTable->AlterRecordL(aWPtr,aRPtr,aLength,iInlineLimit);
   936 	}
   937 
   938 TInt CDbStoreTable::CAlter::RecordExpansion(const TUint8*,TInt)
   939 //
   940 // providing for CClusterCache::MAlter
   941 // just return the maximum possible expansion, rather than a record-specific one
   942 //
   943 	{
   944 	return iExpansion;
   945 	}