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".
8 // Initial Contributors:
9 // Nokia Corporation - initial contribution.
18 // Class CDbStoreDatabase::CCompactor
20 NONSHARABLE_CLASS(CDbStoreDatabase::CCompactor) : public CDbStoreDatabase::CStepper
23 static CCompactor* NewL(CDbDatabase::TUtility aType, CStreamStore& aStore,
24 TInt& aReclaim, TInt& aStep);
27 inline CCompactor(TInt& aReclaim);
29 TInt StepL(TInt aStep);
31 RStoreReclaim iReclaimer;
35 inline CDbStoreDatabase::CCompactor::CCompactor(TInt& aReclaim)
40 CDbStoreDatabase::CCompactor* CDbStoreDatabase::CCompactor::NewL(CDbDatabase::TUtility aType,
42 TInt& aReclaim,TInt& aStep)
44 CCompactor* self=new(ELeave) CCompactor(aReclaim);
45 CleanupStack::PushL(self);
46 if (aType==CDbDatabase::ECompact)
47 self->iReclaimer.CompactL(aStore,aStep);
49 self->iReclaimer.OpenL(aStore,aStep);
54 CDbStoreDatabase::CCompactor::~CCompactor()
60 // Single step the compactor
61 // We cannot deal with the "in use" scenario as we could end up locking out forever
62 // that has to be left to clients using the RDbIncremental interface
64 TInt CDbStoreDatabase::CCompactor::StepL(TInt aStep)
66 iReclaimer.NextL(aStep);
69 iReclaim=iReclaimer.Available();
76 // Class CDbStoreDatabase
78 EXPORT_C CDbStoreDatabase::CDbStoreDatabase()
79 :iReclaim(KErrGeneral)
84 // Create a StoreDatabase object. This type shares the store
86 CDbStoreDatabase* CDbStoreDatabase::NewLC(CStreamStore* aStore)
89 CDbStoreDatabase* self=new(ELeave) CDbStoreDatabase;
90 CleanupStack::PushL(self);
97 //SYMBIAN_REMOVE_TRIVIAL_ENCRYPTION version of the method.
98 CDbDatabase* CDbStoreDatabase::CreateL(CStreamStore* aStore,TStreamId& aStreamId)
100 CDbStoreDatabase* self=NewLC(aStore);
101 aStreamId=self->ConstructL();
102 CDbDatabase* db=self->InterfaceL();
103 CleanupStack::Pop(); // self
107 // SYMBIAN_REMOVE_TRIVIAL_ENCRYPTION version of the method.
108 // Phase 2 construction for creating a new database
109 // Initialise a new database object, creating the required persistent structure
111 EXPORT_C TStreamId CDbStoreDatabase::ConstructL()
113 __ASSERT(iStore); // this must have been provided by now
114 iVersion=TUint8(EDbStoreVersion2);
116 iPagePool->Create(Store());
117 iTokenId=Store().ExtendL();
119 iSchemaId=Store().ExtendL();
125 // SYMBIAN_REMOVE_TRIVIAL_ENCRYPTION version of the method.
126 // Open phase #2: Authenticate the client
128 EXPORT_C void CDbStoreDatabase::AuthenticateL()
132 // first client to open the database, so complete initialisation now
138 // SYMBIAN_REMOVE_TRIVIAL_ENCRYPTION version of the method.
139 void CDbStoreDatabase::InitPagePoolL()
141 iPagePool = new(ELeave) RStorePagePool;
145 CDbSource* CDbStoreDatabase::OpenL(CStreamStore* aStore,TStreamId aStreamId)
147 CDbStoreDatabase* self=NewLC(aStore);
148 self->RestoreL(aStreamId);
149 CDbSource* src=self->SourceL();
150 CleanupStack::Pop(); // self
155 // Phase 2 construction for opening a database
156 // Client must still authenticate before it can be used
158 EXPORT_C void CDbStoreDatabase::RestoreL(TStreamId aStreamId)
160 __ASSERT(iStore); // this must have been provided by now
162 // read the databse header for encryption information
163 RStoreReadStream strm;
164 strm.OpenLC(Store(),aStreamId);
166 CleanupStack::PopAndDestroy(); // strm
170 // Load the root stream header (up to the security key)
172 void CDbStoreDatabase::ReadHeaderL(RReadStream& aStream)
176 if (uid!=KDbmsStoreDatabase)
177 __LEAVE(KErrArgument);
181 case EDbStoreCompressed:
182 aStream>>CompressionL();
184 case EDbStoreVersion2:
187 __LEAVE(KErrNotSupported);
192 CDbStoreCompression& CDbStoreDatabase::CompressionL()
194 CDbStoreCompression* c=iCompression;
196 iFilter=iCompression=c=CDbStoreCompression::NewL();
200 EXPORT_C CDbStoreDatabase::~CDbStoreDatabase()
204 iPagePool->Release();
207 delete iClusterCache;
215 // Validate the column set first
217 EXPORT_C CDbTableDef* CDbStoreDatabase::CreateTableL(const TDesC& aName,const CDbColSet& aColSet,const CDbKey* aPrimaryKey)
220 __LEAVE(KErrNotSupported); // Store database does not support primary keys
221 CDbStoreDef* def=CDbStoreDef::NewLC(aName,aColSet);
222 def->SetTokenId(CDbStoreRecords::CreateL(ClusterCacheL()));
227 EXPORT_C CDbTableIndexDef* CDbStoreDatabase::CreateIndexL(const CDbTableDef& aTable,const TDesC& aName,const CDbKey& aKey)
229 CDbStoreIndexDef* def=CDbStoreIndexDef::NewLC(aName,aKey,aTable.Columns());
230 def->SetTokenId(CDbStoreIndex::CreateL(*this,*def));
231 CleanupStack::Pop(); // IndexDef
236 // Destroy the entire database...
238 EXPORT_C void CDbStoreDatabase::DestroyL()
240 iPagePool->Discard();
241 iPagePool->ReclaimAllL(); // reclaim all page pool space
242 iStore->DeleteL(iSchemaId);
243 iStore->DeleteL(iTokenId);
247 EXPORT_C CDbTable* CDbStoreDatabase::TableL(const CDbTableDef& aDef)
249 return new(ELeave) CDbStoreTable(*this,aDef);
253 // load the schema for the database
255 EXPORT_C void CDbStoreDatabase::LoadSchemaL()
257 RDbStoreReadStream strm(*this);
258 strm.OpenLC(Store(),iSchemaId);
260 strm.FilterL(strm.EMixed,iSchemaId.Value());
262 RDbTableSchema& schema=Schema();
265 for (TInt ii=tables;ii>0;--ii)
266 schema.Add(CDbStoreDef::NewL(strm));
267 CleanupStack::PopAndDestroy();
268 strm.OpenLC(Store(),iTokenId);
269 strm>>iFlags>>iPoolToken;
270 iPagePool->Open(Store(),iPoolToken);
271 CleanupStack::PopAndDestroy();
275 // Re-write the schema stream
277 void CDbStoreDatabase::ReplaceSchemaL()
279 RDbStoreWriteStream out(*this);
280 out.ReplaceLC(Store(),iSchemaId);
281 out<<KDbmsStoreDatabase<<iVersion;
284 case EDbStoreCompressed:
285 __ASSERT(iCompression);
288 case EDbStoreVersion2:
293 out.FilterL(out.EMixed,iSchemaId.Value());
295 TSglQueIterC<CDbStoreDef> iter(Schema());
299 out<<TCardinality(count);
301 for (const CDbStoreDef* def;(def=iter++)!=0;)
304 CleanupStack::PopAndDestroy();
308 // Re-write the token stream, removing the mark
310 void CDbStoreDatabase::ReplaceTokenL(TUint aFlags)
312 RStoreWriteStream out;
313 out.ReplaceLC(Store(),iTokenId);
314 out<<TUint8(aFlags&EDamaged)<<iPagePool->Token();
316 CleanupStack::PopAndDestroy();
320 // Return some database property
322 EXPORT_C TInt CDbStoreDatabase::Property(CDbDatabase::TProperty aProperty)
326 case CDbDatabase::EIsDamaged:
327 return iFlags&EDamaged ? 1 : 0;
328 case CDbDatabase::ECompactable:
331 return CDbTableDatabase::Property(aProperty);
336 // mark the database as dirty
338 void CDbStoreDatabase::MarkL()
340 if (!(iFlags&EModified))
342 RStoreWriteStream out;
343 out.OpenLC(Store(),iTokenId);
344 out.WriteUint8L(EDamaged); // mark as dirty
348 CleanupStack::PopAndDestroy();
354 // Reset all cache buffers
356 EXPORT_C void CDbStoreDatabase::Idle()
366 delete iClusterCache;
372 // Commit the store, and when all is well, clear the token
374 void CDbStoreDatabase::SynchStoreL(TDbLockType aLock)
378 TUint newflags=iFlags&~EModified;
379 if (aLock==EDbRecoveryLock)
381 if (aLock==EDbRecoveryLock || iFlags&EModified)
382 ReplaceTokenL(newflags);
383 if (aLock>=EDbWriteLock || iSharedStore)
386 iFlags=TUint8(newflags);
387 iPoolToken=iPagePool->Token();
392 // An index has been successfully recovered, commit it
394 void CDbStoreDatabase::IndexRecoveredL()
396 SynchStoreL(EDbSchemaLock);
400 // Ensure all data is in the store
402 EXPORT_C void CDbStoreDatabase::SynchL(TDbLockType aLock)
404 if (aLock==EDbSchemaLock)
407 iClusterCache->FlushL();
412 // Unwind the store, throw out changes, etc
414 EXPORT_C void CDbStoreDatabase::Revert(TDbLockType aLock)
416 if (aLock>=EDbWriteLock)
419 iClusterCache->Discard();
422 if (iFlags&EModified)
424 iPagePool->Open(Store(),iPoolToken); // reset the page pool
426 else if (!iSharedStore) // don't touch the store if not shared
432 // Ensure we have a cluster cache and return it
434 CClusterCache& CDbStoreDatabase::ClusterCacheL()
436 CClusterCache* cache=iClusterCache;
438 iClusterCache=cache=CClusterCache::NewL(*this);
443 // Ensure we have a page cache and return the pool
445 MPagePool& CDbStoreDatabase::PagePoolL()
449 iPageCache=CPageCache::NewL(EPageCachePages);
450 iPagePool->Set(*iPageCache);
456 // Create an incremental object that compacts the store
458 EXPORT_C CDbTableDatabase::CStepper* CDbStoreDatabase::UtilityL(CDbDatabase::TUtility aType,TInt& aStep)
462 case CDbDatabase::EStats:
463 case CDbDatabase::ECompact:
464 return CCompactor::NewL(aType,Store(),iReclaim,aStep);
465 case CDbDatabase::ERecover:
466 return RecoverL(aStep);
468 return CDbTableDatabase::UtilityL(aType,aStep);
473 // Create an incremental object to destroy a table
475 EXPORT_C CDbTableDatabase::CStepper* CDbStoreDatabase::RecordDiscarderL(const CDbTableDef& aTable,TInt& aStep)
477 CDbStoreTable::CDiscarder* discarder=new(ELeave) CDbStoreTable::CDiscarder;
478 CleanupStack::PushL(discarder);
479 aStep=discarder->OpenL((CDbStoreTable*)TableL(aTable));
485 // Create an incremental object to destroy an index
487 EXPORT_C CDbTableDatabase::CStepper* CDbStoreDatabase::IndexDiscarderL(const CDbTableDef& aTable,const CDbTableIndexDef& anIndex,TInt& aStep)
489 CDbStoreIndex::CDiscarder* discarder=new(ELeave) CDbStoreIndex::CDiscarder;
490 CleanupStack::PushL(discarder);
491 CDbStoreIndex* index=CDbStoreIndex::NewL(*this,(const CDbStoreIndexDef&)anIndex,aTable);
492 aStep=discarder->Open(index);
499 // Provide a stepper to alter the table data
500 // if no data to alter, return 0
502 EXPORT_C CDbTableDatabase::CStepper* CDbStoreDatabase::TableAlterL(CDbTableDef& aTable,const HDbColumnSet& aNewSet,TInt& aStep)
504 CDbStoreDef& def=STATIC_CAST(CDbStoreDef&,aTable);
506 aStep=CDbStoreRecords::CardinalityL(Store(),def);
508 return NULL; // no data to modify
510 // check that all added columns are nullable
511 HDbColumnSet::TIteratorC col=aNewSet.Begin();
512 HDbColumnSet::TIteratorC end=aNewSet.End();
515 if (col->iFlags&TDbColumnDef::EAdded && col->iAttributes&TDbCol::ENotNull)
516 __LEAVE(KErrArgument); // added column is not nullable
519 // check to see if anything is being dropped or changed type
520 col=aTable.Columns().Begin();
521 end=aTable.Columns().End();
522 while (!(col->iFlags&(TDbColumnDef::EDropped|TDbColumnDef::EChangedType|TDbColumnDef::EChangedLen)))
525 { // no changes which affect layout, so no work required
531 CDbStoreTable::CAlter* alter=new(ELeave) CDbStoreTable::CAlter;
532 CleanupStack::PushL(alter);
533 alter->OpenL((CDbStoreTable*)TableL(def),aNewSet);
538 EXPORT_C void CDbStoreDatabase::Reserved_1()
542 EXPORT_C void CDbStoreDatabase::Reserved_2()