os/persistentdata/persistentstorage/dbms/udbms/UD_ROW.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 "UD_STD.H"
    17 
    18 #define UNUSED_VAR(a) a = a
    19 
    20 // Class TDbColumn
    21 
    22 EXPORT_C void TDbColumn::SetL(const TAny* aPtr,TInt aSize)
    23 	{
    24 	Mem::Copy(Row().SetColumnWidthL(iColumn,aSize),aPtr,aSize);
    25 	}
    26 
    27 EXPORT_C void TDbColumn::SetL(TUint32 aValue)
    28 	{
    29 	*(TUint32*)Row().SetColumnWidthL(iColumn,sizeof(aValue))=aValue;
    30 	}
    31 
    32 EXPORT_C void TDbColumn::SetL(TInt64 aValue)
    33 	{
    34 	*(TInt64*)Row().SetColumnWidthL(iColumn,sizeof(aValue))=aValue;
    35 	}
    36 
    37 EXPORT_C void TDbColumn::SetL(TReal32 aValue) __SOFTFP
    38 	{
    39 	*(TReal32*)Row().SetColumnWidthL(iColumn,sizeof(aValue))=aValue;
    40 	}
    41 
    42 EXPORT_C void TDbColumn::SetL(TReal64 aValue) __SOFTFP
    43 	{
    44 	*(TReal64*)Row().SetColumnWidthL(iColumn,sizeof(aValue))=aValue;
    45 	}
    46 
    47 EXPORT_C void TDbColumn::SetL(const TDesC8& aValue)
    48 	{
    49 	SetL(aValue.Ptr(),aValue.Length());
    50 	}
    51 
    52 EXPORT_C void TDbColumn::SetL(const TDesC16& aValue)
    53 	{
    54 	SetL(aValue.Ptr(),aValue.Length()<<1);
    55 	}
    56 
    57 EXPORT_C void TDbColumn::SetBlobL(TDbBlobId aBlobId,TInt aSize)
    58 	{
    59 	new(Row().SetColumnWidthL(iColumn,TDbBlob::RefSize())) TDbBlob(aBlobId,aSize);
    60 	}
    61 
    62 EXPORT_C void TDbColumn::SetBlobL(const TUint8* aData,TInt aSize)
    63 	{
    64 	new(Row().SetColumnWidthL(iColumn,TDbBlob::InlineSize(aSize))) TDbBlob(aData,aSize);
    65 	}
    66 
    67 EXPORT_C TDbBlob& TDbColumn::InitBlobL()
    68 	{
    69 	return *new(Row().SetColumnWidthL(iColumn,sizeof(TDbBlob))) TDbBlob;
    70 	}
    71 
    72 EXPORT_C void TDbColumn::CommitBlob(const TDbBlob& aBlob)
    73 	{
    74 #if defined(_DEBUG)
    75 	TDbColumnC colC(*this);
    76 	__ASSERT(&colC.Blob()==&aBlob);
    77 	__ASSERT(colC.Size()>=aBlob.CellSize());
    78 #endif
    79 	TRAPD(errCode, Row().SetColumnWidthL(iColumn,aBlob.Size() ? aBlob.CellSize() : 0));
    80     UNUSED_VAR(errCode);
    81 	}
    82 
    83 // The row buffer
    84 // Minimum data width is 4 bytes for strict alignment
    85 // No integral set members for small integers and extractors do the cast down from 4-byte integers
    86 
    87 EXPORT_C RDbRow::RDbRow()
    88 //
    89 // default to dynamic-owned empty row buffer
    90 //
    91 	:iFirst(0),iLast(0),iEnd(0),iCell(0),iColumn(TUint(EOwned))
    92 	{}
    93 
    94 EXPORT_C void RDbRow::Open(TAny* aBuf,TInt aSize,TInt aMaxSize)
    95 //
    96 // Use the provided row buffer
    97 //
    98 	{
    99 	iFirst=(TDbCell*)aBuf;
   100 	iEnd=PtrAdd(iFirst,aMaxSize);
   101 	iColumn=0;			// non-owned buffer
   102 	SetSize(aSize);
   103 	}
   104 
   105 EXPORT_C void RDbRow::CreateL(TInt aMaxSize)
   106 	{
   107 	__ASSERT(!iFirst && Owned());
   108 	GrowL(aMaxSize);
   109 	}
   110 
   111 void RDbRow::PushL()
   112 //
   113 // Dynamic reallocation requires that the buffer is pushed
   114 //
   115 	{
   116 	CleanupClosePushL(*this);
   117 	}
   118 
   119 EXPORT_C void RDbRow::Close()
   120 //
   121 // release the buffer if owned, and return to default constructed state
   122 //
   123 	{
   124 	if (Owned())
   125 		User::Free(iFirst);
   126 	iFirst=iLast=iEnd=iCell=0;
   127 	iColumn=TUint(EOwned);
   128 	}
   129 
   130 EXPORT_C void RDbRow::SetSize(TInt aSize)
   131 //
   132 // Set the current buffer size: assert that it is valid
   133 //
   134 	{
   135 	__ASSERT(PtrAdd(iFirst,aSize)<=iEnd);
   136 	iLast=PtrAdd(iFirst,aSize);
   137 	SetCache(iFirst,0);
   138 	}
   139 
   140 EXPORT_C void RDbRow::GrowL(TInt aMaxSize)
   141 //
   142 // Grow the buffer to at least aMaxSize, the buffer empty on return
   143 //
   144 	{
   145 	if (aMaxSize>MaxSize())
   146 		{
   147 		ReallocL(aMaxSize);
   148 		Reset();
   149 	//
   150 		__ASSERT(iFirst<=iEnd);
   151 		__ASSERT(iFirst<=iLast && iLast<=iEnd);
   152 		__ASSERT(iFirst<=iCell && iCell<=iLast);
   153 		}
   154 	}
   155 
   156 void RDbRow::ExtendL(TInt aAdjust)
   157 //
   158 // Adjust the buffer length by aAdjust, leave if no room, preserve contents and cache
   159 //
   160 	{
   161 	TDbCell* newLast=PtrAdd(iLast,aAdjust);
   162 	if (newLast>iEnd)
   163 		{
   164 		TInt move=ReallocL(Diff(iFirst,newLast));
   165 		iCell=PtrAdd(iCell,move);
   166 		newLast=PtrAdd(newLast,move);
   167 		}
   168 	iLast=newLast;
   169 // check invariant
   170 	__ASSERT(iFirst<=iEnd);
   171 	__ASSERT(iFirst<=iLast && iLast<=iEnd);
   172 	__ASSERT(iFirst<=iCell && iCell<=iLast);
   173 	}
   174 
   175 TInt RDbRow::ReallocL(TInt aMaxSize)
   176 //
   177 // Grow the buffer to aMaxSize, return the offset of buffer movement
   178 // leave iLast and cache untouched
   179 //
   180 	{
   181 	__ASSERT(aMaxSize>MaxSize());
   182 	if (!Owned())
   183 		__LEAVE(KErrTooBig);	// cannot reallocate if not owned
   184 //
   185 	aMaxSize+=EGranularity-1;
   186 	aMaxSize&=~(EGranularity-1);
   187 	TDbCell* buf=(TDbCell*)User::ReAllocL(iFirst,aMaxSize);
   188 	TInt move=Diff(iFirst,buf);
   189 	iFirst=buf;
   190 	iEnd=PtrAdd(buf,aMaxSize);
   191 	return move;
   192 	}
   193 
   194 TDbCell* RDbRow::Column(TInt aColIx) const
   195 //
   196 // aCol is now zero-based
   197 // Return a pointer to column aCol in the buffer
   198 // Return 0 if the column is past the end
   199 //
   200 	{
   201 	TInt ix=Column();
   202 	TDbCell* pC;
   203 	if (aColIx<ix)
   204 		{
   205 		ix=0;
   206 		pC=iFirst;
   207 		}
   208 	else
   209 		pC=iCell;
   210 	TDbCell* last=iLast;
   211 	for (;;)
   212 		{
   213 		if (pC>=last)
   214 			break;
   215 		if (ix>=aColIx)
   216 			break;
   217 		pC=pC->Next();
   218 		++ix;
   219 		}
   220 	CONST_CAST(RDbRow*,this)->SetCache(pC,ix);
   221 	return pC<last?pC:0;
   222 	}
   223 
   224 LOCAL_C const TDbCell* NullCell()
   225 	{
   226 	static TUint32 const NullCell[3]={0,0,0};	// 3 zero words represent any "Null" value
   227 //
   228 	return REINTERPRET_CAST(const TDbCell*,&NullCell[0]);
   229 	}
   230 
   231 EXPORT_C const TDbCell* RDbRow::ColCell(TDbColNo aColNo) const
   232 	{
   233 	__ASSERT(aColNo>0);
   234 	const TDbCell* cell=Column(aColNo-1);
   235 	return cell && cell->Length() ? cell : NullCell();
   236 	}
   237 
   238 EXPORT_C TAny* RDbRow::SetColumnWidthL(TDbColNo aColNo,TInt aWidth)
   239 // set the width for column aCol to Width
   240 // add extra NULL columns to buffer as required
   241 // return pointer to data for that column
   242 	{
   243 	__ASSERT(aColNo>0);
   244 	TDbCell* pC=Column(--aColNo);
   245 	if (pC==0)
   246 		{		// add extra NULL columns to buffer
   247 		if (aWidth==0)
   248 			return 0;	// setting to NULL, don't bother padding
   249 		TInt cFill=(aColNo-iColumn)*sizeof(TInt);
   250 		ExtendL(cFill+TDbCell::Size(aWidth));
   251 		pC=iCell;
   252 		Mem::FillZ(pC,cFill);
   253 		pC=PtrAdd(pC,cFill);		// set cache
   254 		SetCache(pC,aColNo);
   255 		}
   256 	else
   257 		{
   258 		TInt adjust=TDbCell::Size(aWidth)-pC->Size();	// how much to add
   259 		if (adjust!=0)
   260 			{
   261 			ExtendL(adjust);
   262 			pC=iCell;		// may have moved in extension
   263 			TDbCell* pNext=pC->Next();
   264 			TDbCell* pAdjust=PtrAdd(pNext,adjust);
   265 			Mem::Move(pAdjust,pNext,Diff(pAdjust,iLast));
   266 			}
   267 		}
   268 	pC->SetLength(aWidth);
   269 	return pC->Data();
   270 	}