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 #define UNUSED_VAR(a) a = a
20 const TUint KTableExpiry=0x100000; // ~1.0s
24 NONSHARABLE_CLASS(CDbBlobCleanup) : public CArrayFixFlat<TDbBlobId>
27 static CDbBlobCleanup* NewLC(CDbBlobSpace& aBlobSpace);
30 inline CDbBlobCleanup(CDbBlobSpace& aBlobSpace);
32 CDbBlobSpace& iBlobSpace;
35 inline CDbBlobCleanup::CDbBlobCleanup(CDbBlobSpace& aBlobSpace)
36 : CArrayFixFlat<TDbBlobId>(8),iBlobSpace(aBlobSpace)
39 CDbBlobCleanup* CDbBlobCleanup::NewLC(CDbBlobSpace& aBlobSpace)
41 CDbBlobCleanup* self=new(ELeave) CDbBlobCleanup(aBlobSpace);
42 CleanupStack::PushL(self);
46 CDbBlobCleanup::~CDbBlobCleanup()
51 const TDbBlobId* blob=&(*this)[0];
52 const TDbBlobId* const end=blob+count;
53 for (;blob<end;++blob)
54 iBlobSpace.Delete(*blob);
61 EXPORT_C CDbTable::CDbTable(CDbTableDatabase& aDatabase,const CDbTableDef& aDef)
62 : iDatabase(&aDatabase),iDef(&aDef)
65 aDatabase.AddTable(*this); // we reference database
68 EXPORT_C CDbTable::~CDbTable()
73 __ASSERT(!InUse()); // cannot be directly deleted
78 void CDbTable::Disconnect()
80 // Disconnect the table from the database collection.
84 Database().RemoveTable(*this);
85 TRAPD(errCode, ApplyToComponentsL(CDbRecordBase::DoDelete));
94 { // were idle or cached
97 Cache().Release(*this); // was cached
104 void CDbTable::Close()
106 // We may destroy this object when the last reference goes away
113 delete this; // disconnected table
116 CDbTableDatabase& db=Database();
117 if (!db.Transaction().IsLocked())
118 Idle(); // no transaction, idle now
119 db.Close(); // this must be done last to avoid early self destruction
124 void CDbTable::Idle()
126 // Called when idle, change to cached state
130 __ASSERT(IsActive());
133 Cache().Hold(this,KTableExpiry); // may delete this
136 void CDbTable::FlushL()
138 // Ensure all records objects are flushed
141 __ASSERT(IsActive());
143 ApplyToComponentsL(CDbRecordBase::DoFlushL);
146 void CDbTable::Abandon()
148 // Discard all components
151 __ASSERT(IsActive());
152 TRAPD(errCode, ApplyToComponentsL(CDbRecordBase::DoAbandon));
154 iIndexesEnd=NULL; // flags indexes as abandoned
158 void CDbTable::Release()
160 // Release the table and all its cursors as DDL is about to begin
163 __ASSERT(IsActive());
167 Cache().Release(*this);
168 // fall throught to Idle
176 iDatabase=0; // this marks us as released
182 void CDbTable::ApplyToBlobsL(RDbRow& aRow,TBlobFuncL aFuncL,CDbBlobCleanup* aCleanup)
184 __ASSERT(Def().Columns().HasLongColumns());
185 CDbBlobSpace* blobs=BlobsL();
188 HDbColumnSet::TIteratorC iter=Def().Columns().Begin();
189 const HDbColumnSet::TIteratorC end=Def().Columns().End();
192 if (!TDbCol::IsLong(iter->Type()))
194 const TDbColumnC column(aRow,col);
197 aFuncL(*blobs,CONST_CAST(TDbBlob&,column.Blob()),iter->Type(),aCleanup);
198 } while (++col,++iter<end);
201 LOCAL_C void DuplicateBlobL(CDbBlobSpace& aBlobStore,TDbBlob& aBlob,TDbColType aType,CDbBlobCleanup* aCleanup)
204 if (aBlob.IsInline())
206 // need to duplicate blob
207 RReadStream old(aBlobStore.ReadLC(aBlob.Id(),aType));
208 TDbBlobId& newId=aCleanup->ExtendL();
210 RWriteStream dup(aBlobStore.CreateL(newId,aType));
212 dup.WriteL(old,aBlob.Size());
214 CleanupStack::PopAndDestroy(2); // old and dup streams
215 aBlob.SetId(newId); // row is writable
218 void CDbTable::DuplicateBlobsL(RDbRow& aRow)
220 // duplicate any blobs
223 if (!Def().Columns().HasLongColumns())
226 CDbBlobCleanup* cleaner=CDbBlobCleanup::NewLC(*BlobsL());
227 ApplyToBlobsL(aRow,DuplicateBlobL,cleaner);
229 CleanupStack::PopAndDestroy();
232 TBool CDbTable::ExistsL(TDbRecordId aRecordId)
234 // Check that aRecordId is good for this table
237 __ASSERT(IsActive() && InUse());
238 return RecordsL().ExistsL(aRecordId);
241 void CDbTable::NewRowL(RDbRow& aRow)
243 // Initialise any auto-increment columns in the row
246 const HDbColumnSet& columns=Def().Columns();
247 if (!columns.HasAutoIncrement())
250 TUint value=RecordsL().AutoIncrementL();
252 HDbColumnSet::TIteratorC iter=columns.Begin();
253 const HDbColumnSet::TIteratorC end=columns.End();
256 if (iter->iAttributes&TDbCol::EAutoIncrement)
258 // auto-increment only for integral types <=32 bits wide
259 __ASSERT(iter->iType<=EDbColUint32);
260 TDbColumn column(aRow,col);
261 column.SetL(TUint32(value));
263 } while (++col,++iter<end);
266 void CDbTable::ValidateL(const RDbRow& aRow)
268 // Ensure that the column data conforms to type size/flags etc
271 HDbColumnSet::TIteratorC iter=Def().Columns().Begin();
272 const HDbColumnSet::TIteratorC end=Def().Columns().End();
273 const TDbCell* const last=aRow.Last();
274 for (const TDbCell* column=aRow.First();column<last;++iter,column=column->Next())
276 TInt size=column->Length();
279 if (iter->iAttributes&TDbCol::ENotNull)
281 __LEAVE(KErrNotFound);
286 const TUint32* data=(const TUint32*)column->Data();
291 __LEAVE(KErrOverflow);
297 __LEAVE(KErrOverflow);
303 if (TInt16(val)!=val)
304 __LEAVE(KErrOverflow);
310 if (TUint8(val)!=val)
311 __LEAVE(KErrOverflow);
317 if (TUint16(val)!=val)
318 __LEAVE(KErrOverflow);
325 if (iter->iMaxLength==KDbUndefinedLength)
327 if (size>iter->iMaxLength)
328 __LEAVE(KErrOverflow);
330 case EDbColLongBinary:
331 case EDbColLongText8:
332 case EDbColLongText16:
333 if (iter->iMaxLength==KDbUndefinedLength)
335 size=((TDbBlob*)data)->Size();
336 if (size==KDbUndefinedLength)
338 if (iter->iType==EDbColText16)
340 if (size>iter->iMaxLength)
341 __LEAVE(KErrOverflow);
347 for (;iter<end;++iter)
349 if (iter->iAttributes&TDbCol::ENotNull)
351 __LEAVE(KErrNotFound);
357 void CDbTable::ReadRowL(RDbRow& aRow,TDbRecordId aRecordId)
359 // Read a record from the table
362 CopyToRowL(aRow,RecordsL().ReadL(aRecordId));
365 void CDbTable::PrepareAppendL(const RDbTableRow& aRow)
367 // Validate a new record for appending
372 CDbRecordIndex** end=iIndexesEnd;
373 for (CDbRecordIndex** pix=iIndexes;pix<end;++pix)
375 CDbRecordIndex& ix=**pix;
378 if (ix.FindL(KDbNullRecordId,aRow)==CDbRecordIndex::EKeyMatch)
379 __LEAVE(KErrAlreadyExists); // duplicate found
383 TDbRecordId CDbTable::AppendRowL(const RDbTableRow& aRow)
385 // Validate and add a new record to the table and any open indexes
388 CDbRecordSpace& records=RecordsL();
389 CopyFromRow(records.NewL(RecordLength(aRow)),aRow);
390 TDbRecordId id=records.AppendL();
391 CDbRecordIndex** end=iIndexesEnd;
392 for (CDbRecordIndex** pix=iIndexes;pix<end;++pix)
394 CDbRecordIndex& ix=**pix;
397 __DEBUG(TInt dbgchk=) ix.InsertL(id,aRow);
404 void CDbTable::PrepareReplaceL(const RDbTableRow& aRow,TDbRecordId aRecordId)
406 // Validate a record for replacement
412 CDbRecordIndex** end=iIndexes;
413 for (CDbRecordIndex** pix=iIndexesEnd;--pix>=end;)
416 CDbRecordIndex& ix=**pix;
419 switch (ix.FindL(aRecordId,aRow))
421 case CDbRecordIndex::ENoMatch: // key has changed in index
424 case CDbRecordIndex::EKeyMatch: // duplicate found
425 __LEAVE(KErrAlreadyExists);
426 case CDbRecordIndex::EEntryMatch: // no change in index
433 void CDbTable::DoReplaceRowL(const RDbRow& aRow,TDbRecordId aRecordId)
435 CopyFromRow(RecordsL().ReplaceL(aRecordId,RecordLength(aRow)),aRow);
439 void CDbTable::ReplaceRowL(RDbTableRow& aRow,TDbRecordId aRecordId)
441 // Replace a record in the table
444 if (Def().Columns().HasLongColumns())
445 CheckInliningL(aRow);
446 TUint32 update=iUpdateMap;
449 DoReplaceRowL(aRow,aRecordId);
452 RDbTableRow oldRow; // temporary row buffer for old row values
454 oldRow.PushL(); // cleanup buffer if there is trouble
455 ReadRowL(oldRow,aRecordId);
456 DoReplaceRowL(aRow,aRecordId);
457 for (CDbRecordIndex** pix=iIndexes;update;++pix,update>>=1)
461 CDbRecordIndex& index=**pix;
462 index.DeleteL(aRecordId,oldRow);
463 __DEBUG(TInt dbgchk=) index.InsertL(aRecordId,aRow);
467 CleanupStack::PopAndDestroy(); // temp row buffer
470 LOCAL_C void CheckInlineL(CDbBlobSpace& aBlobStore,TDbBlob& aBlob,TDbColType aType,CDbBlobCleanup*)
472 if (!aBlob.IsInline())
474 if (aBlob.Size()>aBlobStore.InlineLimit())
475 aBlob.SetId(aBlobStore.CreateL(aType,aBlob.Data(),aBlob.Size()));
478 void CDbTable::CheckInliningL(RDbRow& aRow)
480 // Ensure that all Blobs are within the current inline limit
483 ApplyToBlobsL(aRow,CheckInlineL);
486 LOCAL_C void DiscardBlobL(CDbBlobSpace& aBlobStore,TDbBlob& aBlob,TDbColType,CDbBlobCleanup*)
488 if (!aBlob.IsInline())
489 aBlobStore.DeleteL(aBlob.Id());
492 EXPORT_C void CDbTable::DiscardBlobsL(RDbRow& aRow)
494 // Default implemtation xlates the record and then walks the row buffer
497 ApplyToBlobsL(aRow,DiscardBlobL);
500 void CDbTable::DeleteRowL(RDbTableRow& aRow,TDbRecordId aRecordId)
502 // Delete the record from the file and unlock it.
507 if (Def().Columns().HasLongColumns())
509 // Read data from the stream but do not delete the stream yet.
510 aRow.ReadL(aRecordId);
513 CDbRecordIndex** end=iIndexes;
514 CDbRecordIndex** pix=iIndexesEnd;
516 aRow.ReadL(aRecordId);
517 RecordsL().EraseL(aRecordId);
520 CDbRecordIndex& ix=**pix;
522 ix.DeleteL(aRecordId,aRow);
525 if (Def().Columns().HasLongColumns())
527 // Now delete the stream.
534 EXPORT_C CDbRecordSpace& CDbTable::RecordsL()
536 __ASSERT(IsActive() && InUse());
537 CDbRecordSpace* rec=iRecords;
539 iRecords=rec=RecordSpaceL();
541 __LEAVE(KErrCorrupt);
545 EXPORT_C CDbBlobSpace* CDbTable::BlobsL()
547 __ASSERT(IsActive() && InUse());
548 CDbBlobSpace* blob=iBlobs;
550 iBlobs=blob=BlobSpaceL();
552 __LEAVE(KErrCorrupt);
556 EXPORT_C CDbRecordIndex& CDbTable::IndexL(const CDbTableIndexDef& aIndex)
558 // Load the index associated with the index definition and ensure it is operational
561 __ASSERT(IsActive() && InUse());
562 // find the matching slot in the indexes array
563 CDbRecordIndex** slot=&iIndexes[0];
564 for (TSglQueIterC<CDbTableIndexDef> iter(Def().Indexes().AsQue());iter++!=&aIndex;)
566 __ASSERT(iIndexesEnd==NULL||(slot>=iIndexes&&slot<iIndexesEnd));
567 // load (if required) and open the index
568 CDbRecordIndex* index=*slot;
570 *slot=index=RecordIndexL(aIndex);
572 __LEAVE(KErrCorrupt);
576 void CDbTable::EnsureIndexesL()
578 // Ensure that all indexes are open
581 __ASSERT(IsActive() && InUse());
582 if (iIndexesEnd==NULL)
584 CDbRecordIndex** pp=iIndexes;
585 TSglQueIterC<CDbTableIndexDef> iter(Def().Indexes().AsQue());
586 for (const CDbTableIndexDef* xDef;(xDef=iter++)!=NULL;++pp)
588 CDbRecordIndex* ix=*pp;
590 *pp=ix=RecordIndexL(*xDef);
591 ix->OpenL(); // ignore broken-ness
597 CDbRecordIter* CDbTable::IteratorL()
599 return RecordsL().IteratorL();
602 CDbRecordIter* CDbTable::IteratorL(const CDbTableIndexDef& aIndex,TUint aInclusion,const TDbLookupKey* aLowerBound,const TDbLookupKey* aUpperBound)
604 // create an interator for the index parameter
607 return IndexL(aIndex).IteratorL(aInclusion,aLowerBound,aUpperBound);
610 EXPORT_C TInt CDbTable::IndexSpanL(const CDbTableIndexDef&,TUint,const TDbLookupKey*,const TDbLookupKey*)
612 // Default implementation: no statistics are available
615 return EUnavailableSpan;
618 CDbRecordIter* CDbTable::IteratorL(const TDesC& aIndex)
620 // create an interator for the index named
623 return IteratorL(Def().Indexes().FindL(aIndex));
626 void CDbTable::ApplyToComponentsL(void (*anOperationL)(CDbRecordBase*))
628 // Invoke anOperation on all components of the table
632 anOperationL(iRecords);
634 anOperationL(iBlobs);
635 CDbRecordIndex** const ixs=iIndexes;
636 CDbRecordIndex** pix=iIndexesEnd;
638 pix=&iIndexes[KDbTableMaxIndexes];
644 // Class CDbtable::TValid
646 // this class is used by the cursor to check that it is still operational
648 CDbTable::TValid::TValid(CDbTable& aTable)
651 __ASSERT(aTable.IsActive());
652 iRollback.Construct(aTable.Database().Transaction().RollbackGeneration());
655 TBool CDbTable::TValid::Reset()
657 TBool b=Table().IsActive();
663 void CDbTable::TValid::CheckL() const
665 CDbTableDatabase* d=Table().iDatabase;
667 __LEAVE(KErrDisconnected);
669 d->Transaction().ReadyL();
670 if (iRollback.Changed())
671 __LEAVE(KErrNotReady);