os/persistentdata/persistentstorage/dbms/sdbms/SD_CURS.CPP
author sl@SLION-WIN7.fritz.box
Fri, 15 Jun 2012 03:10:57 +0200
changeset 0 bde4ae8d615e
permissions -rw-r--r--
First public contribution.
     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 // DBMS server proxy cursor
    15 // 
    16 //
    17 
    18 #include "SD_STD.H"
    19 
    20 const TDbBlobId KUnknownBlobId=~0u;
    21 
    22 // Class HDbsColumns
    23 
    24 class HDbsColumns
    25 	{
    26 public:
    27 	static HDbsColumns* NewL(TInt aCount);
    28 	inline void Set(const TUint8* aPtr);
    29 	inline TPtr8 Ptr();
    30 	inline TInt Count() const;
    31 	inline TDbColType Type(TDbColNo aCol) const;
    32 	void Check(TDbColNo aCol) const;
    33 private:
    34 	inline HDbsColumns(TInt aCount);
    35 private:
    36 	TInt iCount;
    37 	TUint8 iType[1];
    38 	};
    39 
    40 inline HDbsColumns::HDbsColumns(TInt aCount)
    41 	:iCount(aCount)
    42 	{}
    43 inline void HDbsColumns::Set(const TUint8* aPtr)
    44 	{Mem::Copy(iType,aPtr,iCount);}
    45 inline TPtr8 HDbsColumns::Ptr()
    46 	{return TPtr8(iType,iCount);}
    47 inline TInt HDbsColumns::Count() const
    48 	{return iCount;}
    49 inline TDbColType HDbsColumns::Type(TDbColNo aCol) const
    50 	{__DEBUG(Check(aCol));return TDbColType(iType[aCol-1]);}
    51 
    52 HDbsColumns* HDbsColumns::NewL(TInt aCount)
    53 	{
    54 	return new(User::AllocL(_FOFF(HDbsColumns,iType[aCount]))) HDbsColumns(aCount);
    55 	}
    56 
    57 void HDbsColumns::Check(TDbColNo aColNo) const
    58 	{
    59 	__ASSERT_ALWAYS(aColNo>0&&aColNo<=iCount,Panic(EDbsInvalidColumn));
    60 	}
    61 
    62 // Class CDbsCursor
    63 
    64 CDbsCursor::CDbsCursor()
    65 	:iState(EAtBeginning)
    66 	{}
    67 
    68 CDbsCursor* CDbsCursor::NewL(const RDbsObject& aDbs,TDbsFunction aFunction,TIpcArgs& aArgs)
    69 	{
    70 	CDbsCursor* self=new(ELeave) CDbsCursor;
    71 	CleanupStack::PushL(self);
    72 	self->ConstructL(aDbs,aFunction,aArgs);
    73 	CleanupStack::Pop();
    74 	return self;
    75 	}
    76 
    77 void CDbsCursor::ConstructL(const RDbsObject& aDbs,TDbsFunction aFunction,TIpcArgs& aArgs)
    78 //
    79 // construct a cursor from the supplied parameters
    80 // Retrieve the column types, in one pass if possible
    81 //
    82 	{
    83 	TPckgBuf<TDbsColumns> cols;
    84 	aArgs.Set(3,&cols);
    85 	iObject.OpenL(aDbs,aFunction,aArgs);
    86 	iColumns=HDbsColumns::NewL(cols().iCount);
    87 	if (cols().iCount<=cols().EMax)
    88 		iColumns->Set(cols().iData);	// we have the column types
    89 	else
    90 		{	// too many for the fixed buffer, make a second call to get the columns
    91 		TPtr8 ptr=iColumns->Ptr();
    92 		aArgs.Set(3,&ptr);
    93 		iObject.SendReceiveL(EDbsCursorColumnTypes,aArgs);
    94 		}
    95 	}
    96 
    97 CDbsCursor::~CDbsCursor()
    98 	{
    99 	iObject.Close();
   100 	delete iColumns;
   101 	iRow.Close();
   102 	}
   103 
   104 TDbColType CDbsCursor::Type(TDbColNo aCol) const
   105 	{
   106 	iColumns->Check(aCol);
   107 	return iColumns->Type(aCol);
   108 	}
   109 
   110 void CDbsCursor::Reset()
   111 	{
   112 	iState=EAtBeginning;
   113 	iObject.SendReceive(EDbsCursorReset);
   114 	}
   115 
   116 TBool CDbsCursor::EvaluateL()
   117 	{
   118 	iState=EUnknown;
   119 	return iObject.SendReceiveL(EDbsCursorEvaluate);
   120 	}
   121 
   122 void CDbsCursor::Evaluate(TRequestStatus& aStatus)
   123 	{
   124 	iState=EUnknown;
   125 	iObject.SendReceive(EDbsCursorEvaluate,aStatus);
   126 	}
   127 
   128 TBool CDbsCursor::Unevaluated()
   129 	{
   130 	return iObject.SendReceive(EDbsCursorUnevaluated);
   131 	}
   132 
   133 void CDbsCursor::SetIndexL(const TDesC* anIndex)
   134 	{
   135 	iObject.SendReceiveL(EDbsCursorSetIndex,TIpcArgs(anIndex));
   136 	iState=EAtBeginning;
   137 	}
   138 
   139 TBool CDbsCursor::SeekL(const TDbLookupKey& aKey,RDbTable::TComparison aComparison)
   140 	{
   141 	TPtrC8 key(TDbsParam::PrepareLC(aKey));
   142 	iState=EUnknown;
   143 	TBool found=iObject.SendReceiveL(EDbsCursorSeek,TIpcArgs(&key,key.Length(),aComparison));
   144 	CleanupStack::PopAndDestroy();	// key
   145 	if (found)
   146 		iState=EAtRow;
   147 	return found;
   148 	}
   149 
   150 TBool CDbsCursor::AtBeginning()
   151 	{
   152 	if (iState==EUnknown && iObject.SendReceive(EDbsCursorAtBeginning)>0)
   153 		iState=EAtBeginning;
   154 	return iState==EAtBeginning;
   155 	}
   156 
   157 TBool CDbsCursor::AtEnd()
   158 	{
   159 	if (iState==EUnknown && iObject.SendReceive(EDbsCursorAtEnd)>0)
   160 		iState=EAtEnd;
   161 	return iState==EAtEnd;
   162 	}
   163 
   164 TBool CDbsCursor::AtRow()
   165 	{
   166 	if (iState==EUnknown && iObject.SendReceive(EDbsCursorAtRow)>0)
   167 		iState=EAtRow;
   168 	return iState>=EAtRow;
   169 	}
   170 
   171 TInt CDbsCursor::CountL(RDbRowSet::TAccuracy aAccuracy)
   172 //
   173 // valid response is >=0 + undefined (==-1)
   174 // server returns +1 for non-error conditions
   175 //
   176 	{
   177 	return iObject.SendReceiveL(EDbsCursorCount,TIpcArgs(aAccuracy))-1;
   178 	}
   179 
   180 TBool CDbsCursor::GotoL(RDbRowSet::TPosition aPosition)
   181 	{
   182 	TBool atrow=RetrieveL(EDbsCursorGotoPos,aPosition);
   183 	if (atrow)
   184 		iState=ERetrieve;
   185 	else
   186 		{
   187 		switch (aPosition)
   188 			{
   189 		default:
   190 			__ASSERT(0);
   191 		case RDbRowSet::EEnd:
   192 		case RDbRowSet::EFirst:
   193 		case RDbRowSet::ENext:
   194 			iState=EAtEnd;
   195 			break;
   196 		case RDbRowSet::EBeginning:
   197 		case RDbRowSet::ELast:
   198 		case RDbRowSet::EPrevious:
   199 			iState=EAtBeginning;
   200 			break;
   201 			}
   202 		}
   203 	return atrow;
   204 	}
   205 
   206 void CDbsCursor::Bookmark(TDbBookmark::TMark& aMark)
   207 	{
   208 	TPckg<TDbBookmark::TMark> pckg(aMark);
   209 	iObject.SendReceive(EDbsCursorBookmark,TIpcArgs(TIpcArgs::ENothing,TIpcArgs::ENothing,TIpcArgs::ENothing,&pckg));
   210 	}
   211 
   212 void CDbsCursor::GotoL(const TDbBookmark::TMark& aMark)
   213 	{
   214 	TPckgC<TDbBookmark::TMark> pckg(aMark);
   215 	iState=EUnknown;
   216 	iObject.SendReceiveL(EDbsCursorGotoBookmark,TIpcArgs(&pckg));
   217 	}
   218 
   219 LOCAL_C void CancelCursor(TAny* aPtr)
   220 	{
   221 	STATIC_CAST(CDbCursor*,aPtr)->Cancel();
   222 	}
   223 
   224 TBool CDbsCursor::RetrieveL(TDbsFunction aFunction,TInt aArg0)
   225 //
   226 // Read the row buffer
   227 //
   228 	{
   229 	iState=EUnknown;
   230 	TInt size;
   231 	for (;;)
   232 		{
   233 		TInt max=iRow.MaxSize();
   234 		TPtr8 row((TUint8*)iRow.First(),max);
   235 		size=iObject.SendReceiveL(aFunction,TIpcArgs(aArg0,TIpcArgs::ENothing,max,&row))-1;
   236 		if (size<0)
   237 			return EFalse;		// no row to retrieve
   238 		if (size<=max)
   239 			break;
   240 		// didn't fit! Grow the buffer
   241 		if (aFunction!=EDbsCursorRetrieveRow)
   242 			{
   243 			CleanupStack::PushL(TCleanupItem(CancelCursor,this));	// in case this goes wrong!
   244 			aFunction=EDbsCursorRetrieveRow;
   245 			}
   246 		iRow.GrowL(size);
   247 		}
   248 	iRow.SetSize(size);
   249 	if (aFunction==EDbsCursorRetrieveRow)
   250 		CleanupStack::Pop();
   251 	return ETrue;
   252 	}
   253 
   254 void CDbsCursor::GetL()
   255 	{
   256 	if (iState!=ERetrieve)
   257 		RetrieveL(EDbsCursorGet);
   258 	iState=ERead;
   259 	}
   260 
   261 void CDbsCursor::InsertL(TInsert aClearRow)
   262 	{
   263 	RetrieveL(EDbsCursorInsert,aClearRow);
   264 	iState=EWrite;
   265 	iChangedBlob=EFalse;
   266 	}
   267 
   268 void CDbsCursor::UpdateL()
   269 	{
   270 	RetrieveL(EDbsCursorUpdate);
   271 	iState=EWrite;
   272 	iChangedBlob=EFalse;
   273 	}
   274 
   275 void CDbsCursor::Cancel()
   276 	{
   277 	if (iState==EUnknown || iState==EWrite)
   278 		iObject.SendReceive(EDbsCursorCancel);
   279 	}
   280 
   281 void CDbsCursor::PutL()
   282 	{
   283 	TInt size=iRow.Size();
   284 	TPtrC8 row((const TUint8*)iRow.First(),size);
   285 	iObject.SendReceiveL(EDbsCursorPut,TIpcArgs(&row,size,iChangedBlob));// if iChangedBlob false, server can optimize put
   286 	iState=ERead;		// we can still look at the row we put
   287 	}
   288 
   289 void CDbsCursor::DeleteL()
   290 	{
   291 	iState=EUnknown;
   292 	iObject.SendReceiveL(EDbsCursorDelete);
   293 	}
   294 
   295 TInt CDbsCursor::ColumnCount()
   296 	{
   297 	return iColumns->Count();
   298 	}
   299 
   300 void CDbsCursor::ColumnsL(CDbColSet& aColSet)
   301 //
   302 // optimise the retreival of all columns
   303 //
   304 	{
   305 	TIpcArgs m;
   306 	RReadStream in(HDbsBuf::NewLC(iObject,EDbsCursorColumns,m));
   307 	in>>aColSet;
   308 	CleanupStack::PopAndDestroy();	// buffer
   309 	}
   310 
   311 void CDbsCursor::ColumnDef(TDbCol& aCol,TDbColNo aColNo)
   312 	{
   313 	TPckg<TDbCol> pckg(aCol);
   314 	iColumns->Check(aColNo);
   315 	iObject.SendReceive(EDbsCursorColumnDef,TIpcArgs(aColNo,TIpcArgs::ENothing,TIpcArgs::ENothing,&pckg));
   316 	}
   317 
   318 TDbColType CDbsCursor::ColumnType(TDbColNo aCol)
   319 	{
   320 	return Type(aCol);
   321 	}
   322 
   323 RDbRow* CDbsCursor::RowBuffer()
   324 //
   325 // Invoked by the server-- should not be called
   326 //
   327 	{
   328 	__ASSERT(0);
   329 	return 0;
   330 	}
   331 
   332 TDbColumnC CDbsCursor::ColumnC(TDbColNo aCol)
   333 //
   334 // check row is valid for extraction
   335 //
   336 	{
   337 	__ASSERT_ALWAYS(iState>=ERead,Panic(EDbsNoRowData));
   338 	return TDbColumnC(iRow,aCol);
   339 	}
   340 
   341 TDbColumn CDbsCursor::Column(TDbColNo aCol)
   342 //
   343 // check row is valid for writing
   344 //
   345 	{
   346 	__ASSERT_ALWAYS(iState==EWrite,Panic(EDbsNotInUpdate));
   347 	return TDbColumn(iRow,aCol);
   348 	}
   349 
   350 void CDbsCursor::ReplaceBlobL(TDbColumn& aCol)
   351 //
   352 // We no longer know what the shape of a blob is, mark it as "unknown"
   353 //
   354 	{
   355 	iChangedBlob=ETrue;
   356 	aCol.SetBlobL(KUnknownBlobId);
   357 	}
   358 
   359 void CDbsCursor::SetNullL(TDbColNo aCol)
   360 //
   361 // Make the column Null
   362 //
   363 	{
   364 	TDbColumn col=Column(aCol);
   365 	if (!TDbCol::IsLong(Type(aCol)))
   366 		col.SetNull();
   367 	else if (!TDbColumnC(col).IsNull())
   368 		{
   369 		ReplaceBlobL(col);
   370 		iObject.SendReceiveL(EDbsCursorSetNull,TIpcArgs(aCol));
   371 		}
   372 	}
   373 
   374 TInt CDbsCursor::ColumnSize(TDbColNo aCol)
   375 	{
   376 	TDbColumnC col(ColumnC(aCol));
   377 	if (!TDbCol::IsLong(Type(aCol)))
   378 		return col.Size();
   379 	TInt size=col.Blob().Size();
   380 	if (size>=0)
   381 		return size;
   382 	// unknown size, so ask the server
   383 	return iObject.SendReceive(EDbsCursorColumnSize,TIpcArgs(aCol));
   384 	}
   385 
   386 MStreamBuf* CDbsCursor::ColumnSourceL(TDbColNo aCol)
   387 	{
   388 	TDbColumnC col(ColumnC(aCol));
   389 	if (!TDbCol::IsLong(iColumns->Type(aCol)))
   390 		return HDbsReadBuf::NewL(col.PtrC8());
   391 	// blobs
   392 	const TDbBlob& blob=col.Blob();
   393 	if (blob.IsInline())
   394 		return HDbsReadBuf::NewL(blob.PtrC8());
   395 	// get it from the server
   396 	TIpcArgs args(aCol);
   397 	return HDbsBuf::NewL(iObject,EDbsCursorColumnSource,args);
   398 	}
   399 
   400 MStreamBuf* CDbsCursor::ColumnSinkL(TDbColNo aCol)
   401 	{
   402 	TDbColumn col(Column(aCol));
   403 	ReplaceBlobL(col);
   404 	TIpcArgs args(aCol);
   405 	return HDbsBuf::NewL(iObject,EDbsCursorColumnSink,args);
   406 	}
   407 
   408 CDbRowConstraint* CDbsCursor::OpenConstraintL(const TDbQuery& aQuery)
   409 	{
   410 	const TDesC& des = aQuery.Query();
   411 	CDbsConstraint* c=new(ELeave) CDbsConstraint();
   412 	CleanupStack::PushL(c);
   413 	c->iObject.OpenL(iObject,EDbsCursorOpenConstraint,TIpcArgs(&des,aQuery.Comparison()));
   414 	CleanupStack::Pop();
   415 	return c;
   416 	}
   417 
   418 TBool CDbsCursor::MatchL(CDbRowConstraint& aConstraint)
   419 	{
   420 	TInt handle = STATIC_CAST(CDbsConstraint&,aConstraint).iObject.Handle();
   421 	return iObject.SendReceiveL(EDbsCursorMatch,TIpcArgs(handle));
   422 	}
   423 
   424 TInt CDbsCursor::FindL(RDbRowSet::TDirection aDirection,const TDbQuery& aCriteria)
   425 //
   426 // Server returns result-KErrNotFound, to ensure non-leaving on not found
   427 //
   428 	{
   429 	const TDesC& des = aCriteria.Query();
   430 	iState=EUnknown;
   431 	TInt f=iObject.SendReceiveL(EDbsCursorFind,TIpcArgs(&des,aCriteria.Comparison(),aDirection))+KErrNotFound;
   432 	if (f>=0)
   433 		iState=EAtRow;
   434 	return f;
   435 	}
   436 
   437 // Class CDbsConstraint
   438 
   439 CDbsConstraint::~CDbsConstraint()
   440 	{
   441 	iObject.Close();
   442 	}