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.
17 #include "U32STD_DBMS.H"
19 #define UNUSED_VAR(a) a = a
21 const TUint KTableExpiry=0x100000; // ~1.0s
25 NONSHARABLE_CLASS(CDbBlobCleanup) : public CArrayFixFlat<TDbBlobId>
28 static CDbBlobCleanup* NewLC(CDbBlobSpace& aBlobSpace);
31 inline CDbBlobCleanup(CDbBlobSpace& aBlobSpace);
33 CDbBlobSpace& iBlobSpace;
36 inline CDbBlobCleanup::CDbBlobCleanup(CDbBlobSpace& aBlobSpace)
37 : CArrayFixFlat<TDbBlobId>(8),iBlobSpace(aBlobSpace)
40 CDbBlobCleanup* CDbBlobCleanup::NewLC(CDbBlobSpace& aBlobSpace)
42 CDbBlobCleanup* self=new(ELeave) CDbBlobCleanup(aBlobSpace);
43 CleanupStack::PushL(self);
47 CDbBlobCleanup::~CDbBlobCleanup()
52 const TDbBlobId* blob=&(*this)[0];
53 const TDbBlobId* const end=blob+count;
54 for (;blob<end;++blob)
55 iBlobSpace.Delete(*blob);
62 EXPORT_C CDbTable::CDbTable(CDbTableDatabase& aDatabase,const CDbTableDef& aDef)
63 : iDatabase(&aDatabase),iDef(&aDef)
66 aDatabase.AddTable(*this); // we reference database
69 EXPORT_C CDbTable::~CDbTable()
74 __ASSERT(!InUse()); // cannot be directly deleted
79 void CDbTable::Disconnect()
81 // Disconnect the table from the database collection.
85 Database().RemoveTable(*this);
86 TRAPD(errCode, ApplyToComponentsL(CDbRecordBase::DoDelete));
95 { // were idle or cached
98 Cache().Release(*this); // was cached
105 void CDbTable::Close()
107 // We may destroy this object when the last reference goes away
114 delete this; // disconnected table
117 CDbTableDatabase& db=Database();
118 if (!db.Transaction().IsLocked())
119 Idle(); // no transaction, idle now
120 db.Close(); // this must be done last to avoid early self destruction
125 void CDbTable::Idle()
127 // Called when idle, change to cached state
131 __ASSERT(IsActive());
134 Cache().Hold(this,KTableExpiry); // may delete this
137 void CDbTable::FlushL()
139 // Ensure all records objects are flushed
142 __ASSERT(IsActive());
144 ApplyToComponentsL(CDbRecordBase::DoFlushL);
147 void CDbTable::Abandon()
149 // Discard all components
152 __ASSERT(IsActive());
153 TRAPD(errCode, ApplyToComponentsL(CDbRecordBase::DoAbandon));
155 iIndexesEnd=NULL; // flags indexes as abandoned
159 void CDbTable::Release()
161 // Release the table and all its cursors as DDL is about to begin
164 __ASSERT(IsActive());
168 Cache().Release(*this);
169 // fall throught to Idle
177 iDatabase=0; // this marks us as released
183 void CDbTable::ApplyToBlobsL(RDbRow& aRow,TBlobFuncL aFuncL,CDbBlobCleanup* aCleanup)
185 __ASSERT(Def().Columns().HasLongColumns());
186 CDbBlobSpace* blobs=BlobsL();
189 HDbColumnSet::TIteratorC iter=Def().Columns().Begin();
190 const HDbColumnSet::TIteratorC end=Def().Columns().End();
193 if (!TDbCol::IsLong(iter->Type()))
195 const TDbColumnC column(aRow,col);
198 aFuncL(*blobs,CONST_CAST(TDbBlob&,column.Blob()),iter->Type(),aCleanup);
199 } while (++col,++iter<end);
202 LOCAL_C void DuplicateBlobL(CDbBlobSpace& aBlobStore,TDbBlob& aBlob,TDbColType aType,CDbBlobCleanup* aCleanup)
205 if (aBlob.IsInline())
207 // need to duplicate blob
208 RReadStream old(aBlobStore.ReadLC(aBlob.Id(),aType));
209 TDbBlobId& newId=aCleanup->ExtendL();
211 RWriteStream dup(aBlobStore.CreateL(newId,aType));
213 dup.WriteL(old,aBlob.Size());
215 CleanupStack::PopAndDestroy(2); // old and dup streams
216 aBlob.SetId(newId); // row is writable
219 void CDbTable::DuplicateBlobsL(RDbRow& aRow)
221 // duplicate any blobs
224 if (!Def().Columns().HasLongColumns())
227 CDbBlobCleanup* cleaner=CDbBlobCleanup::NewLC(*BlobsL());
228 ApplyToBlobsL(aRow,DuplicateBlobL,cleaner);
230 CleanupStack::PopAndDestroy();
233 TBool CDbTable::ExistsL(TDbRecordId aRecordId)
235 // Check that aRecordId is good for this table
238 __ASSERT(IsActive() && InUse());
239 return RecordsL().ExistsL(aRecordId);
242 void CDbTable::NewRowL(RDbRow& aRow)
244 // Initialise any auto-increment columns in the row
247 const HDbColumnSet& columns=Def().Columns();
248 if (!columns.HasAutoIncrement())
251 TUint value=RecordsL().AutoIncrementL();
253 HDbColumnSet::TIteratorC iter=columns.Begin();
254 const HDbColumnSet::TIteratorC end=columns.End();
257 if (iter->iAttributes&TDbCol::EAutoIncrement)
259 // auto-increment only for integral types <=32 bits wide
260 __ASSERT(iter->iType<=EDbColUint32);
261 TDbColumn column(aRow,col);
262 column.SetL(TUint32(value));
264 } while (++col,++iter<end);
267 void CDbTable::ValidateL(const RDbRow& aRow)
269 // Ensure that the column data conforms to type size/flags etc
272 HDbColumnSet::TIteratorC iter=Def().Columns().Begin();
273 const HDbColumnSet::TIteratorC end=Def().Columns().End();
274 const TDbCell* const last=aRow.Last();
275 for (const TDbCell* column=aRow.First();column<last;++iter,column=column->Next())
277 TInt size=column->Length();
280 if (iter->iAttributes&TDbCol::ENotNull)
282 __LEAVE(KErrNotFound);
287 const TUint32* data=(const TUint32*)column->Data();
292 __LEAVE(KErrOverflow);
298 __LEAVE(KErrOverflow);
304 if (TInt16(val)!=val)
305 __LEAVE(KErrOverflow);
311 if (TUint8(val)!=val)
312 __LEAVE(KErrOverflow);
318 if (TUint16(val)!=val)
319 __LEAVE(KErrOverflow);
326 if (iter->iMaxLength==KDbUndefinedLength)
328 if (size>iter->iMaxLength)
329 __LEAVE(KErrOverflow);
331 case EDbColLongBinary:
332 case EDbColLongText8:
333 case EDbColLongText16:
334 if (iter->iMaxLength==KDbUndefinedLength)
336 size=((TDbBlob*)data)->Size();
337 if (size==KDbUndefinedLength)
339 if (iter->iType==EDbColText16)
341 if (size>iter->iMaxLength)
342 __LEAVE(KErrOverflow);
348 for (;iter<end;++iter)
350 if (iter->iAttributes&TDbCol::ENotNull)
352 __LEAVE(KErrNotFound);
358 void CDbTable::ReadRowL(RDbRow& aRow,TDbRecordId aRecordId)
360 // Read a record from the table
363 CopyToRowL(aRow,RecordsL().ReadL(aRecordId));
366 void CDbTable::PrepareAppendL(const RDbTableRow& aRow)
368 // Validate a new record for appending
373 CDbRecordIndex** end=iIndexesEnd;
374 for (CDbRecordIndex** pix=iIndexes;pix<end;++pix)
376 CDbRecordIndex& ix=**pix;
379 if (ix.FindL(KDbNullRecordId,aRow)==CDbRecordIndex::EKeyMatch)
380 __LEAVE(KErrAlreadyExists); // duplicate found
384 TDbRecordId CDbTable::AppendRowL(const RDbTableRow& aRow)
386 // Validate and add a new record to the table and any open indexes
389 CDbRecordSpace& records=RecordsL();
390 CopyFromRow(records.NewL(RecordLength(aRow)),aRow);
391 TDbRecordId id=records.AppendL();
392 CDbRecordIndex** end=iIndexesEnd;
393 for (CDbRecordIndex** pix=iIndexes;pix<end;++pix)
395 CDbRecordIndex& ix=**pix;
398 __DEBUG(TInt dbgchk=) ix.InsertL(id,aRow);
405 void CDbTable::PrepareReplaceL(const RDbTableRow& aRow,TDbRecordId aRecordId)
407 // Validate a record for replacement
413 CDbRecordIndex** end=iIndexes;
414 for (CDbRecordIndex** pix=iIndexesEnd;--pix>=end;)
417 CDbRecordIndex& ix=**pix;
420 switch (ix.FindL(aRecordId,aRow))
422 case CDbRecordIndex::ENoMatch: // key has changed in index
425 case CDbRecordIndex::EKeyMatch: // duplicate found
426 __LEAVE(KErrAlreadyExists);
427 case CDbRecordIndex::EEntryMatch: // no change in index
434 void CDbTable::DoReplaceRowL(const RDbRow& aRow,TDbRecordId aRecordId)
436 CopyFromRow(RecordsL().ReplaceL(aRecordId,RecordLength(aRow)),aRow);
440 void CDbTable::ReplaceRowL(RDbTableRow& aRow,TDbRecordId aRecordId)
442 // Replace a record in the table
445 if (Def().Columns().HasLongColumns())
446 CheckInliningL(aRow);
447 TUint32 update=iUpdateMap;
450 DoReplaceRowL(aRow,aRecordId);
453 RDbTableRow oldRow; // temporary row buffer for old row values
455 oldRow.PushL(); // cleanup buffer if there is trouble
456 ReadRowL(oldRow,aRecordId);
457 DoReplaceRowL(aRow,aRecordId);
458 for (CDbRecordIndex** pix=iIndexes;update;++pix,update>>=1)
462 CDbRecordIndex& index=**pix;
463 index.DeleteL(aRecordId,oldRow);
464 __DEBUG(TInt dbgchk=) index.InsertL(aRecordId,aRow);
468 CleanupStack::PopAndDestroy(); // temp row buffer
471 LOCAL_C void CheckInlineL(CDbBlobSpace& aBlobStore,TDbBlob& aBlob,TDbColType aType,CDbBlobCleanup*)
473 if (!aBlob.IsInline())
475 if (aBlob.Size()>aBlobStore.InlineLimit())
476 aBlob.SetId(aBlobStore.CreateL(aType,aBlob.Data(),aBlob.Size()));
479 void CDbTable::CheckInliningL(RDbRow& aRow)
481 // Ensure that all Blobs are within the current inline limit
484 ApplyToBlobsL(aRow,CheckInlineL);
487 LOCAL_C void DiscardBlobL(CDbBlobSpace& aBlobStore,TDbBlob& aBlob,TDbColType,CDbBlobCleanup*)
489 if (!aBlob.IsInline())
490 aBlobStore.DeleteL(aBlob.Id());
493 EXPORT_C void CDbTable::DiscardBlobsL(RDbRow& aRow)
495 // Default implemtation xlates the record and then walks the row buffer
498 ApplyToBlobsL(aRow,DiscardBlobL);
501 void CDbTable::DeleteRowL(RDbTableRow& aRow,TDbRecordId aRecordId)
503 // Delete the record from the file and unlock it.
508 if (Def().Columns().HasLongColumns())
510 // Read data from the stream but do not delete the stream yet.
511 aRow.ReadL(aRecordId);
514 CDbRecordIndex** end=iIndexes;
515 CDbRecordIndex** pix=iIndexesEnd;
517 aRow.ReadL(aRecordId);
518 RecordsL().EraseL(aRecordId);
521 CDbRecordIndex& ix=**pix;
523 ix.DeleteL(aRecordId,aRow);
526 if (Def().Columns().HasLongColumns())
528 // Now delete the stream.
535 EXPORT_C CDbRecordSpace& CDbTable::RecordsL()
537 __ASSERT(IsActive() && InUse());
538 CDbRecordSpace* rec=iRecords;
540 iRecords=rec=RecordSpaceL();
542 __LEAVE(KErrCorrupt);
546 EXPORT_C CDbBlobSpace* CDbTable::BlobsL()
548 __ASSERT(IsActive() && InUse());
549 CDbBlobSpace* blob=iBlobs;
551 iBlobs=blob=BlobSpaceL();
553 __LEAVE(KErrCorrupt);
557 EXPORT_C CDbRecordIndex& CDbTable::IndexL(const CDbTableIndexDef& aIndex)
559 // Load the index associated with the index definition and ensure it is operational
562 __ASSERT(IsActive() && InUse());
563 // find the matching slot in the indexes array
564 CDbRecordIndex** slot=&iIndexes[0];
565 for (TSglQueIterC<CDbTableIndexDef> iter(Def().Indexes().AsQue());iter++!=&aIndex;)
567 __ASSERT(iIndexesEnd==NULL||(slot>=iIndexes&&slot<iIndexesEnd));
568 // load (if required) and open the index
569 CDbRecordIndex* index=*slot;
571 *slot=index=RecordIndexL(aIndex);
573 __LEAVE(KErrCorrupt);
577 void CDbTable::EnsureIndexesL()
579 // Ensure that all indexes are open
582 __ASSERT(IsActive() && InUse());
583 if (iIndexesEnd==NULL)
585 CDbRecordIndex** pp=iIndexes;
586 TSglQueIterC<CDbTableIndexDef> iter(Def().Indexes().AsQue());
587 for (const CDbTableIndexDef* xDef;(xDef=iter++)!=NULL;++pp)
589 CDbRecordIndex* ix=*pp;
591 *pp=ix=RecordIndexL(*xDef);
592 ix->OpenL(); // ignore broken-ness
598 CDbRecordIter* CDbTable::IteratorL()
600 return RecordsL().IteratorL();
603 CDbRecordIter* CDbTable::IteratorL(const CDbTableIndexDef& aIndex,TUint aInclusion,const TDbLookupKey* aLowerBound,const TDbLookupKey* aUpperBound)
605 // create an interator for the index parameter
608 return IndexL(aIndex).IteratorL(aInclusion,aLowerBound,aUpperBound);
611 EXPORT_C TInt CDbTable::IndexSpanL(const CDbTableIndexDef&,TUint,const TDbLookupKey*,const TDbLookupKey*)
613 // Default implementation: no statistics are available
616 return EUnavailableSpan;
619 CDbRecordIter* CDbTable::IteratorL(const TDesC& aIndex)
621 // create an interator for the index named
624 return IteratorL(Def().Indexes().FindL(aIndex));
627 void CDbTable::ApplyToComponentsL(void (*anOperationL)(CDbRecordBase*))
629 // Invoke anOperation on all components of the table
633 anOperationL(iRecords);
635 anOperationL(iBlobs);
636 CDbRecordIndex** const ixs=iIndexes;
637 CDbRecordIndex** pix=iIndexesEnd;
639 pix=&iIndexes[KDbTableMaxIndexes];
645 // Class CDbtable::TValid
647 // this class is used by the cursor to check that it is still operational
649 CDbTable::TValid::TValid(CDbTable& aTable)
652 __ASSERT(aTable.IsActive());
653 iRollback.Construct(aTable.Database().Transaction().RollbackGeneration());
656 TBool CDbTable::TValid::Reset()
658 TBool b=Table().IsActive();
664 void CDbTable::TValid::CheckL() const
666 CDbTableDatabase* d=Table().iDatabase;
668 __LEAVE(KErrDisconnected);
670 d->Transaction().ReadyL();
671 if (iRollback.Changed())
672 __LEAVE(KErrNotReady);