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".
8 // Initial Contributors:
9 // Nokia Corporation - initial contribution.
18 // Class CDbTableCursor::HColumns
20 class CDbTableCursor::HColumns
23 static HColumns* NewL(const CDbDataSource* aSource);
24 inline TInt Count() const
26 inline TDbColType Type(TDbColNo aCol) const
27 {__DEBUG(Check(aCol));return TDbColType(iType[aCol-1]);}
28 void Check(TDbColNo aCol) const;
34 CDbTableCursor::HColumns* CDbTableCursor::HColumns::NewL(const CDbDataSource* aSource)
36 TInt count=aSource->ColumnCount();
37 HColumns* self=(HColumns*)User::AllocL(_FOFF(HColumns,iType[count]));
39 TUint8* pp=&self->iType[0];
40 for (TDbColNo ii=1;ii<=count;++ii,++pp)
41 *pp=aSource->ColumnDef(ii).iType;
45 void CDbTableCursor::HColumns::Check(TDbColNo aColNo) const
47 __ASSERT_ALWAYS(TUint(aColNo-1)<TUint(iCount),Panic(EDbInvalidColumn));
50 // Class CDbTableCursor::CConstraint
52 NONSHARABLE_CLASS(CDbTableCursor::CConstraint) : public CDbRowConstraint
55 CConstraint(CDbTableCursor& aCursor,CSqlSearchCondition* aSearchCondition,TDbTextComparison aComparison);
58 inline TBool Check(CDbTableCursor& aCursor) const;
59 inline TBool MatchL() const;
61 CDbTableCursor& iCursor;
62 CSqlSearchCondition* iSearchCondition;
63 const TTextOps& iTextOps;
66 CDbTableCursor::CConstraint::CConstraint(CDbTableCursor& aCursor,CSqlSearchCondition* aSearchCondition,TDbTextComparison aComparison)
67 : iCursor(aCursor), iSearchCondition(aSearchCondition), iTextOps(TTextOps::Ops(aComparison))
70 CDbTableCursor::CConstraint::~CConstraint()
72 delete iSearchCondition;
75 inline TBool CDbTableCursor::CConstraint::Check(CDbTableCursor& aCursor) const
76 {return &iCursor==&aCursor;}
77 inline TBool CDbTableCursor::CConstraint::MatchL() const
78 {return iSearchCondition->EvaluateL(iTextOps);}
80 // Class CDbTableCursor
82 inline void CDbTableCursor::CheckStateL() const
84 inline RDbTransaction& CDbTableCursor::Transaction()
85 {__ASSERT(iValid);return iValid.Transaction();}
86 inline TBool CDbTableCursor::InUpdate() const
87 {return iFlags&(EUpdating|EInserting);}
89 CDbTableCursor::CDbTableCursor(RDbAccessPlan& aPlan,RDbRowSet::TAccess aAccess)
90 : iState(ERowBeginning),iValid(aPlan.Table()),iDataSource(aPlan.Adopt())
96 case RDbRowSet::EUpdatable:
97 iFlags=EUpdatable|EReadable;
99 case RDbRowSet::EReadOnly:
102 case RDbRowSet::EInsertOnly:
108 CDbTableCursor::~CDbTableCursor()
115 CDbTableCursor* CDbTableCursor::NewL(RDbAccessPlan& aPlan,RDbRowSet::TAccess aAccess)
117 CDbTableCursor* self=new(ELeave) CDbTableCursor(aPlan,aAccess);
118 CleanupStack::PushL(self);
119 self->iColumns=HColumns::NewL(self->iDataSource);
125 TDbColType CDbTableCursor::Type(TDbColNo aCol) const
127 iColumns->Check(aCol);
128 return iColumns->Type(aCol);
131 void CDbTableCursor::Reset()
133 // Reset the cursor for re-evaluation
139 iDataSource->Reset();
140 iState=ERowBeginning;
144 TBool CDbTableCursor::EvaluateL()
146 // Do a unit of evaluation work
153 TBool more=iDataSource->EvaluateL(work,iRecord,atRow);
155 { // evaluation results in a record appearing under the cursor
162 case ERowDeletedAtEnd:
163 iState=ERowDeletedAtNext;
172 void CDbTableCursor::Evaluate(TRequestStatus& aStatus)
174 // Asynchronous evaluation: invoke synchronous version
177 TRequestStatus* pStatus=&aStatus;
178 User::RequestComplete(pStatus,CDbCursor::Evaluate());
181 TBool CDbTableCursor::Unevaluated()
183 // Report if there is evaluation to be done
186 return iValid ? iDataSource->Unevaluated() : EFalse;
189 TInt CDbTableCursor::CountL(RDbRowSet::TAccuracy aAccuracy)
193 TInt count=iDataSource->CountL();
194 return (count==KDbUndefinedCount && aAccuracy==RDbRowSet::EEnsure)
195 ? CDbCursor::CountL(aAccuracy)
199 TBool CDbTableCursor::AtBeginning()
201 return iState==ERowBeginning;
204 TBool CDbTableCursor::AtEnd()
206 return iState==ERowEnd;
209 TBool CDbTableCursor::AtRow()
211 return (iState==ERowOK||(iFlags&EInserting));
214 TBool CDbTableCursor::GotoL(RDbRowSet::TPosition aPosition)
216 // Move the cursor in the requested direction
217 // return whether we are at a row or not
227 case RDbRowSet::EFirst:
228 case RDbRowSet::ELast:
230 case RDbRowSet::ENext:
235 case ERowInLimbo: // in between previous and next, must evaluate
238 case ERowBeginning: // goto first record
239 aPosition=RDbRowSet::EFirst;
243 Panic(EDbInvalidRow);
245 case ERowDeletedAtNext: // already have the id
246 if (iDataSource->GotoL(iRecord))
247 { // and the record is still there
252 case ERowDeletedAtEnd: // straight to end
257 case RDbRowSet::EPrevious:
263 case ERowDeletedAtNext: // goto previous will do what we want
264 case ERowInLimbo: // in between previous and next, must evaluate
266 case ERowEnd: // goto last row
267 case ERowDeletedAtEnd: // previous is last row
268 aPosition=RDbRowSet::ELast;
272 Panic(EDbInvalidRow);
276 case RDbRowSet::EBeginning:
277 iState=ERowBeginning;
279 case RDbRowSet::EEnd:
284 switch (iDataSource->GotoL(TDbPosition(aPosition),iRecord))
288 case CDbDataSource::ESynchFailure:
289 __LEAVE(KErrNotReady);
290 case CDbDataSource::ESuccess:
293 case CDbDataSource::ENoRow:
294 iState=TUint8(aPosition<RDbRowSet::EPrevious ? ERowEnd : ERowBeginning);
299 void CDbTableCursor::Bookmark(TDbBookmark::TMark& aMark)
301 // Create a bookmark for the current cursor state
302 // Can bookmark ANY position.
306 aMark.iMark[0]=iState;
307 aMark.iMark[1]=iRecord.Value();
310 void CDbTableCursor::GotoL(const TDbBookmark::TMark& aMark)
312 // Reestablish the cursor state from a bookmark (if possible)
318 iRecord=aMark.iMark[1];
319 TState state=TState(aMark.iMark[0]);
323 Panic(EDbInvalidBookmark);
326 case ERowDeletedAtEnd:
330 case ERowDeletedAtNext:
332 if (!iDataSource->GotoL(iRecord))
333 __LEAVE(KErrNotFound);
336 iState=TUint8(state);
339 void CDbTableCursor::GetL()
341 // read the current row into the row buffer for access
347 iDataSource->ReadRowL(iRecord);
351 void CDbTableCursor::InsertL(TInsert aClearRow)
353 // Insert a new row. If aCLearRow==aCopy, then copy the current row
358 Transaction().DMLPrepareL(*this);
359 if (aClearRow==ECopy)
362 iFlags&=~ERead; // in case of failure in NewRowL
363 iDataSource->NewRowL(iRecord);
366 iDataSource->NewRowL(KDbNullRecordId);
367 iFlags|=EInserting|ERead;
368 Transaction().DMLBegin();
371 void CDbTableCursor::UpdateL()
374 Transaction().DMLPrepareL(*this);
377 Transaction().DMLBegin();
380 void CDbTableCursor::Cancel()
385 RDbTransaction& t=Transaction();
387 t.DMLTouch(); // we've mucked about with BLOBs, so force true roll-back
389 if (iFlags&EUpdating)
391 TRAP_IGNORE(iDataSource->ReadRowL(KDbNullRecordId)); // row buffer contains NULL row (cannot fail)
393 iFlags&=(EUpdatable|EReadable);
397 void CDbTableCursor::PutL()
401 CDbDataSource::TWrite mode=iFlags&EUpdating ? CDbDataSource::EReplace : CDbDataSource::EAppend;
402 iDataSource->PrepareToWriteRowL(mode);
403 RDbTransaction& t=Transaction();
406 iRecord=iDataSource->WriteRowL(mode,iFlags&EReadable ? CDbDataSource::ESynch : CDbDataSource::ENoSynch);
408 if ((iFlags&(EInserting|EReadable))==(EInserting|EReadable))
410 iFlags&=(EUpdatable|EReadable|ERead);
413 void CDbTableCursor::DeleteL()
417 RDbTransaction& t=Transaction();
418 t.DMLPrepareL(*this);
420 CDbDataSource::TDelete del=iDataSource->DeleteRowL(iRecord);
422 CleanupStack::Pop(); // rollback not required
423 iState=TUint8(del+ERowDeletedAtNext);
426 TInt CDbTableCursor::ColumnCount()
428 return iColumns->Count();
431 void CDbTableCursor::ColumnDef(TDbCol& aCol,TDbColNo aColNo)
433 iColumns->Check(aColNo);
435 iDataSource->ColumnDef(aColNo).AsTDbCol(aCol);
438 TDbColType CDbTableCursor::ColumnType(TDbColNo aCol)
443 void CDbTableCursor::ReplaceBlobL(TDbColumn& aCol)
446 const TDbBlob& blob=TDbColumnC(aCol).Blob();
447 if (!blob.IsInline())
450 BlobsL().DeleteL(blob.Id());
454 void CDbTableCursor::AddBlobSource()
456 // Increment the source count and take a read-lock on the database
460 Transaction().ReadBegin(*this);
463 void CDbTableCursor::ReleaseBlobSource()
465 // Decrement the source count and release the read-lock on the database
469 Transaction().ReadRelease(*this);
472 MStreamBuf* CDbTableCursor::ColumnSourceL(TDbColNo aCol)
474 TDbColumnC col(ColumnC(aCol));
475 if ((iFlags&EWriteBuf) || iReadBuf==EMaxReadBuf)
476 __LEAVE(KErrInUse); // only allow 1-write or 255-read streambufs
477 TDbColType type=iColumns->Type(aCol);
478 if (!TDbCol::IsLong(type))
479 return HMemBuf::NewL(*this,col.PtrC8());
481 const TDbBlob& blob=col.Blob();
483 return HMemBuf::NewL(*this,blob.PtrC8());
484 // if small enough, pull the blob data through immediately and avoid locking the database
485 if (blob.Size()<=HHeapBuf::EMaxBlobBuffer)
486 return HHeapBuf::NewL(*this,blob,type);
489 Transaction().ReadPrepareL(*this);
490 HReadBuf* buf=HReadBuf::NewLC(*this);
491 buf->Set(BlobsL().ReadL(blob.Id(),type));
496 MStreamBuf* CDbTableCursor::ColumnSinkL(TDbColNo aCol)
498 TDbColType type=Type(aCol);
499 __ASSERT_ALWAYS(TDbCol::IsLong(type),Panic(EDbWrongType));
500 TDbColumn col=Column(aCol);
503 return HWriteBuf::NewL(*this,col,type);
506 void CDbTableCursor::SetNullL(TDbColNo aCol)
508 // Make the column Null
511 TDbColumn col=Column(aCol);
512 if (TDbCol::IsLong(Type(aCol)))
517 TInt CDbTableCursor::ColumnSize(TDbColNo aCol)
519 TDbColumnC col(ColumnC(aCol));
520 return TDbCol::IsLong(Type(aCol)) ? col.Blob().Size() : col.Size();
523 RDbRow* CDbTableCursor::RowBuffer()
525 // Invoked by the server for whole-row access where possible
528 __ASSERT(iFlags&ERead);
529 return iDataSource->RowBuffer();
532 TDbColumnC CDbTableCursor::ColumnC(TDbColNo aCol)
534 // check row is valid for extraction
537 __ASSERT_ALWAYS(iFlags&ERead,Panic(EDbRowNotRead));
538 return iDataSource->Column(aCol);
541 TDbColumn CDbTableCursor::Column(TDbColNo aCol)
543 // check row is valid for writing
547 return iDataSource->Column(aCol);
550 void CDbTableCursor::SetIndexL(const TDesC* anIndex)
554 iDataSource->SetIndexL(anIndex);
555 iState=ERowBeginning;
558 TBool CDbTableCursor::SeekL(const TDbLookupKey& aKey,RDbTable::TComparison aComparison)
564 TBool atrow=iDataSource->SeekL(aKey,aComparison,iRecord);
570 CDbRowConstraint* CDbTableCursor::OpenConstraintL(const TDbQuery& aCriteria)
572 // Construct a constraint for this rowset
575 CSqlSearchCondition* sc=iDataSource->ParseConstraintLC(aCriteria.Query());
576 CDbRowConstraint* constraint=new(ELeave) CConstraint(*this,sc,aCriteria.Comparison());
581 TBool CDbTableCursor::MatchL(CDbRowConstraint& aConstraint)
583 CConstraint& c=STATIC_CAST(CConstraint&,aConstraint);
584 __ASSERT_ALWAYS(c.Check(*this),Panic(EDbRowSetConstraintMismatch));
589 void CDbTableCursor::CheckReadL() const
591 // Ensure we are a readable cursor
595 if ((iFlags&EReadable)==0)
599 void CDbTableCursor::CheckUpdateL() const
601 // Ensure we are a writable cursor
605 if ((iFlags&EUpdatable)==0)
609 void CDbTableCursor::AssertNoStreams() const
611 __ASSERT_ALWAYS((iFlags&EWriteBuf)==0 && iReadBuf==0,Panic(EDbStreamOpen));
614 void CDbTableCursor::AssertNotInUpdate() const
616 __ASSERT_ALWAYS(!InUpdate(),Panic(EDbInUpdate));
620 void CDbTableCursor::AssertInUpdate() const
622 __ASSERT_ALWAYS(InUpdate(),Panic(EDbNotInUpdate));
626 void CDbTableCursor::AssertValidRow() const
629 __ASSERT_ALWAYS(iState==ERowOK||(iFlags&EInserting),Panic(EDbInvalidRow));