Update contrib.
1 // Copyright (c) 2004-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 #include "SchLogger.h"
23 #define UNUSED_VAR(a) a = a
26 const TInt KMaxChangesBeforeCompact = 5;
31 // -----> SchBackupManagerUtils (header)
34 void SchBackupManagerUtils::Panic(TSchStorePanic aPanic)
36 _LIT(KSchStorePanic, "SchStore");
37 User::Panic(KSchStorePanic, aPanic);
41 // -----> CSchBackupManager (header)
44 CSchBackupManager::CSchBackupManager(RFs& aFsSession)
45 : CActive(EPriorityIdle), iFsSession(aFsSession)
47 // construct backup filename
48 iBackupFileName.Copy(KSchsvrBackupFileName);
49 iBackupFileName[0] = 'A' + static_cast<TInt>(RFs::GetSystemDrive());
51 CActiveScheduler::Add(this);
54 CSchBackupManager::~CSchBackupManager()
56 // This will delete the store and close the compactor
59 delete iScheduleIndex;
63 void CSchBackupManager::ConstructL()
65 iScheduleIndex = CSchScheduleIndex::NewL();
66 iClientIndex = CSchClientIndex::NewL(iFsSession);
70 If this is called during a restore, we want to make sure we want to append
71 to existing structure not always create new object and just append to the queue
72 for example we want to append task to existing client if they have the same name
73 and priority not create another one.
75 void CSchBackupManager::RestoreL(TPriQue<CClientProxy>& aClients,
76 TSglQue<CSchedule>& aTimeSchedules,
77 CSchLogManager& aSchLogManager,TBool aBUR)
79 LOGSTRING("CSchBackupManager::RestoreL");
82 CPermanentFileStore* store = OpenOrCreateBackupStoreLC();
84 // Restore root stream (two pointers to other streams)
85 RStoreReadStream stream;
86 stream.OpenLC(*store, store->Root());
87 stream >> iIndexStreamSchedules;
88 stream >> iIndexStreamClients;
89 CleanupStack::PopAndDestroy(); // stream
92 iClientIndex->RestoreClientsL(aClients, *store, iIndexStreamClients, aSchLogManager,aBUR);
95 iScheduleIndex->RestoreSchedulesL(aTimeSchedules, *store, iIndexStreamSchedules);
98 CleanupStack::PopAndDestroy(store);
101 void CSchBackupManager::PerformStoreOperationL(TSchBackupOperation aAction, const CSchedule& aSchedule)
103 // Perform an schedule-related operation.
106 LOGSTRING3("CSchBackupManager::PerformStoreOperationL - Schedule: %S (%d)", &aSchedule.Name(), aSchedule.Id());
108 if(aSchedule.Persists())
110 // Cancel any compaction that may be going on
113 // Perform the operation
114 CStreamStore* store = OpenOrCreateBackupStoreLC();
115 CleanupRevertPushLC(*store);
119 case ESchBackupOperationAdd:
120 LOGSTRING("CSchBackupManager::PerformStoreOperationL - OpAdd");
121 iScheduleIndex->AddL(iIndexStreamSchedules, *store, aSchedule);
123 case ESchBackupOperationEdit:
124 LOGSTRING("CSchBackupManager::PerformStoreOperationL - OpEdit");
125 iScheduleIndex->EditL(iIndexStreamSchedules, *store, aSchedule);
127 case ESchBackupOperationDelete:
128 LOGSTRING("CSchBackupManager::PerformStoreOperationL - OpDelete");
129 iScheduleIndex->DeleteL(iIndexStreamSchedules, *store, aSchedule);
132 __ASSERT_DEBUG(EFalse, User::Invariant());
133 User::Leave(KErrNotSupported);
135 // Save changes to store
138 CleanupStack::Pop(); // Store reversion cleanup item
139 CleanupStack::PopAndDestroy(store);
141 // Indicate the store has changed and attempt to compact
142 // the store, but only if the required number of store
143 // changes has been met
148 void CSchBackupManager::PerformStoreOperationL(TSchBackupOperation aAction, const CClientProxy& aClient)
150 // Perform an schedule-related operation.
153 LOGSTRING2("CSchBackupManager::PerformStoreOperationL - Client: %S", &aClient.ExecutorFileName());
155 // Cancel any compaction that may be going on
158 // Perform the operation
159 CStreamStore* store = OpenOrCreateBackupStoreLC();
160 CleanupRevertPushLC(*store);
164 case ESchBackupOperationAdd:
165 LOGSTRING("CSchBackupManager::PerformStoreOperationL - OpAdd");
166 iClientIndex->AddL(iIndexStreamClients, *store, aClient);
168 case ESchBackupOperationEdit:
169 LOGSTRING("CSchBackupManager::PerformStoreOperationL - OpEdit");
170 iClientIndex->EditL(iIndexStreamClients, *store, aClient);
172 case ESchBackupOperationDelete:
173 LOGSTRING("CSchBackupManager::PerformStoreOperationL - OpDelete");
174 iClientIndex->DeleteL(iIndexStreamClients, *store, aClient);
177 __ASSERT_DEBUG(EFalse, User::Invariant());
178 User::Leave(KErrNotSupported);
181 // Save changes to store
184 CleanupStack::Pop(); // Store reversion cleanup item
185 CleanupStack::PopAndDestroy(store);
187 // Indicate the store has changed and attempt to compact
188 // the store, but only if the required number of store
189 // changes has been met
193 void CSchBackupManager::RunL()
195 // Perform a store compaction step
198 LOGSTRING("CSchBackupManager::RunL - Performing compaction step");
200 // Is there any more processing required?
201 if (iStoreReclaimerCount() > 0)
203 // Yes, so start next step
204 iStoreReclaimer.Next(iStoreReclaimerCount, iStatus);
206 LOGSTRING("CSchBackupManager::RunL - Requesting another compaction step");
210 // No, we've finised. Clean up previously allocated
212 iStoreReclaimer.Close();
214 TRAPD(errNotRef, iStoreOpenForCompaction->CommitL());
215 UNUSED_VAR(errNotRef);
216 delete iStoreOpenForCompaction;
217 iStoreOpenForCompaction = NULL;
219 // Set this to zero again...
221 LOGSTRING("CSchBackupManager::RunL - Compaction complete");
225 void CSchBackupManager::DoCancel()
227 // Cancel's any asynchronous store compaction
230 LOGSTRING("CSchBackupManager::RunL - Cancelling compaction");
232 iStoreReclaimer.Release();
233 iStoreReclaimer.Close();
235 TRAPD(errNotRef, iStoreOpenForCompaction->CommitL());
236 UNUSED_VAR(errNotRef);
237 delete iStoreOpenForCompaction;
238 iStoreOpenForCompaction = NULL;
241 void CSchBackupManager::StoreChangedL()
243 __ASSERT_DEBUG(!IsActive() && !iStoreOpenForCompaction, User::Invariant());
245 if (RecordStoreChange() >= KMaxChangesBeforeCompact)
248 CStreamStore* store = OpenBackupStoreLC();
250 // Prepare for compaction
252 iStoreReclaimer.CompactL(*store, count);
253 iStoreReclaimerCount = count;
255 // Safe to do this now
256 iStoreOpenForCompaction = store;
259 // Start asynchronous compaction...
260 iStoreReclaimer.Next(iStoreReclaimerCount, iStatus);
265 CPermanentFileStore* CSchBackupManager::OpenBackupStoreLC()
267 return CPermanentFileStore::OpenLC(iFsSession, iBackupFileName, EFileWrite);
270 CPermanentFileStore* CSchBackupManager::OpenOrCreateBackupStoreLC()
272 CPermanentFileStore* store = NULL;
273 TInt error = KErrNone;
275 store = CPermanentFileStore::OpenL(iFsSession, iBackupFileName, EFileWrite);
277 if (error < KErrNone)
279 if (error == KErrNotFound)
281 CreateEmptyBackupL();
282 store = CPermanentFileStore::OpenL(iFsSession, iBackupFileName, EFileWrite);
289 LOGSTRING("CSchBackupManager::OpenOrCreateBackupStoreLC - store opened");
290 CleanupStack::PushL(store);
294 void CSchBackupManager::CleanupRevertPushLC(CStreamStore& aStore)
296 CleanupStack::PushL(TCleanupItem(RevertStore, &aStore));
299 void CSchBackupManager::RevertStore(TAny* aStore)
301 // The Cleanup Item callback function which is used to rever the store
302 // should a leave occur.
305 CStreamStore* store = reinterpret_cast<CStreamStore*>(aStore);
307 LOGSTRING("CSchBackupManager::RevertStore - store reverted");
311 Creates an initialised empty store.
312 The creation process is performed as an atomic operation.
313 If the operation fails somewhere at the middle, CreateEmptyBackupL()
314 will cleanup after itself - the store file will be deleted,
315 iIndexStreamSchedules and iIndexStreamClients stream IDs will be reinitialised
318 void CSchBackupManager::CreateEmptyBackupL()
320 TRAPD(err, DoCreateEmptyBackupL());
324 // If unable to delete file record the fact
325 TInt err2 = iFsSession.Delete(iBackupFileName);
326 if (err2 != KErrNone)
328 LOGSTRING2("CSchBackupManager::CreateEmptyBackupL - File delete error = %d", err2);
330 iIndexStreamSchedules = iIndexStreamClients = KNullStreamId;
336 Creates an initialised empty store.
338 void CSchBackupManager::DoCreateEmptyBackupL()
340 LOGSTRING("CSchBackupManager::CreateEmptyBackupL - trying to create new store");
341 CPermanentFileStore* store = CPermanentFileStore::ReplaceLC(iFsSession, iBackupFileName, EFileWrite);
342 store->SetTypeL(KPermanentFileStoreLayoutUid);
344 // Create emtpy schedule index stream
345 CSchScheduleIndex* indexSchedule = CSchScheduleIndex::NewL();
346 CleanupStack::PushL(indexSchedule);
347 iIndexStreamSchedules = indexSchedule->CreateEmptyIndexL(*store);
348 CleanupStack::PopAndDestroy(indexSchedule);
350 // Create emtpy client index stream
351 CSchClientIndex* indexClient = CSchClientIndex::NewL(iFsSession);
352 CleanupStack::PushL(indexClient);
353 iIndexStreamClients = indexClient->CreateEmptyIndexL(*store);
354 CleanupStack::PopAndDestroy(indexClient);
357 WriteRootStreamL(*store);
361 CleanupStack::PopAndDestroy(store);
362 LOGSTRING("CSchBackupManager::CreateEmptyBackupL - new store created");
365 void CSchBackupManager::WriteRootStreamL(CStreamStore& aStore)
367 // Write the root stream which contains the two stream id's
370 LOGSTRING("CSchBackupManager::WriteRootStreamL - trying to write root stream");
371 RStoreWriteStream stream;
372 TStreamId id = stream.CreateLC(aStore);
374 // This writes a stream id - it doesn't write the actual
375 // dictionary since this is written after every operation.
376 stream << iIndexStreamSchedules;
377 stream << iIndexStreamClients;
381 CleanupStack::PopAndDestroy();// outstream
382 static_cast<CPermanentFileStore&>(aStore).SetRootL(id);
383 LOGSTRING("CSchBackupManager::WriteRootStreamL - root stream written");
387 // -----> CSchScheduleIndex (header)
390 CSchScheduleIndex::CSchScheduleIndex()
394 CSchScheduleIndex* CSchScheduleIndex::NewL()
396 CSchScheduleIndex* self = new(ELeave) CSchScheduleIndex;
400 void CSchScheduleIndex::RestoreSchedulesL(TSglQue<CSchedule>& aTimeSchedules,
401 CStreamStore& aStore,
402 TStreamId aDictionaryStreamId)
404 CSchScheduleDictionary* dictionary = DictionaryLC(aStore, aDictionaryStreamId);
406 // Restore every schedule in the dictionary
407 const TInt count = dictionary->Count();
408 LOGSTRING2("CSchScheduleIndex::RestoreSchedulesL - read %d dictionary entries", count);
410 for(TInt i=0; i<count; i++)
413 TStreamId stream = dictionary->AtIndex(i);
415 // Restore schedule from stream
416 CSchedule* schedule = CSchedule::NewL(static_cast<CFileStore&>(aStore), stream);
419 aTimeSchedules.AddLast(*schedule); // takes ownership
421 LOGSTRING3("CSchScheduleIndex::RestoreSchedulesL - restored schedule: %S, %d", &schedule->Name(), schedule->Id());
423 CleanupStack::PopAndDestroy(dictionary);
426 TStreamId CSchScheduleIndex::CreateEmptyIndexL(CStreamStore& aStore) const
428 CSchScheduleDictionary* dictionary = CSchScheduleDictionary::NewLC();
429 RStoreWriteStream stream;
430 TStreamId id = stream.CreateLC(aStore);
431 stream << *dictionary;
433 CleanupStack::PopAndDestroy(2); // stream, dictionary
437 void CSchScheduleIndex::AddL(TStreamId aIndexStream, CStreamStore& aStore, const CSchedule& aSchedule)
439 // Saves the specified schedule to the store and adds a new entry to the
443 LOGSTRING2("CSchScheduleIndex::AddL - adding a new schedule (id is %d)", aSchedule.Id());
444 RStoreWriteStream stream;
445 TStreamId id = stream.CreateLC(aStore);
448 CleanupStack::PopAndDestroy(); // stream
450 // Read the dictionary and update an entry
451 CSchScheduleDictionary* dictionary = DictionaryLC(aStore, aIndexStream);
452 dictionary->AssignL(aSchedule.Id(), id);
454 // Store the dictionary
455 StoreDictionaryL(aStore, *dictionary, aIndexStream);
456 CleanupStack::PopAndDestroy(dictionary);
457 LOGSTRING("CSchScheduleIndex::AddL - new schedule added okay");
460 void CSchScheduleIndex::EditL(TStreamId aIndexStream, CStreamStore& aStore, const CSchedule& aSchedule)
462 // Replace an existing stream with the contents of aSchedule.
465 // Locate the existing stream (to be replaced)
466 CSchScheduleDictionary* dictionary = DictionaryLC(aStore, aIndexStream);
467 LOGSTRING2("CSchScheduleIndex::EditL - editing schedule with id of %d", aSchedule.Id());
468 TStreamId id = dictionary->At(aSchedule.Id());
469 CleanupStack::PopAndDestroy(dictionary);
470 if (id == KNullStreamId)
472 // Wasn't found, add it instead...
473 LOGSTRING("CSchScheduleIndex::EditL - schedule wasn't found, adding new entry to the store");
474 AddL(aIndexStream, aStore, aSchedule);
478 // Replace - the original stream is orphaned but the new
479 // stream retains the old id.
480 LOGSTRING("CSchScheduleIndex::EditL - replacing original stream");
481 RStoreWriteStream stream;
482 stream.ReplaceLC(aStore, id);
485 CleanupStack::PopAndDestroy(); // stream
489 void CSchScheduleIndex::DeleteL(TStreamId aIndexStream, CStreamStore& aStore, const CSchedule& aSchedule)
491 // Remove an existing schedule from the store.
494 // Locate the existing stream (to be deleted)
495 CSchScheduleDictionary* dictionary = DictionaryLC(aStore, aIndexStream);
496 LOGSTRING2("CSchScheduleIndex::DeleteL - deleting schedule with id of %d", aSchedule.Id());
497 TStreamId id = dictionary->At(aSchedule.Id());
498 if (id == KNullStreamId)
500 LOGSTRING("CSchScheduleIndex::DeleteL - schedule wasn't found! Would panic in debug");
501 __ASSERT_DEBUG(EFalse, SchBackupManagerUtils::Panic(SchBackupManagerUtils::ESchBackupManagerScheduleStreamToDeleteNotFound));
508 // Save changes to store - we have to do this here since
509 // we must assure that the store is fully updated before
510 // we remove this stream from the dictionary (in order to
511 // maintain integrity).
514 // Remove entry from the stream dictionary
515 dictionary->Remove(aSchedule.Id());
516 StoreDictionaryL(aStore, *dictionary, aIndexStream);
518 CleanupStack::PopAndDestroy(dictionary);
521 CSchScheduleIndex::CSchScheduleDictionary* CSchScheduleIndex::DictionaryLC(CStreamStore& aStore, TStreamId aIndexStream)
523 LOGSTRING("CSchScheduleIndex::DictionaryLC - restoring dictionary");
524 CSchScheduleDictionary* dictionary = CSchScheduleDictionary::NewLC();
525 RStoreReadStream stream;
526 stream.OpenLC(aStore, aIndexStream);
527 stream >> *dictionary;
528 CleanupStack::PopAndDestroy(); // stream
529 LOGSTRING("CSchScheduleIndex::DictionaryLC - dictionary restored");
533 void CSchScheduleIndex::StoreDictionaryL(CStreamStore& aStore, const CSchScheduleDictionary& aDictionary, TStreamId aStreamToReplace)
535 LOGSTRING("CSchScheduleIndex::DictionaryLC - storing dictionary");
536 RStoreWriteStream stream;
537 stream.ReplaceLC(aStore, aStreamToReplace);
538 stream << aDictionary;
540 CleanupStack::PopAndDestroy(); // stream
541 LOGSTRING("CSchScheduleIndex::DictionaryLC - dictionary stored");
545 // -----> CSchScheduleDictionary (header)
548 CSchScheduleIndex::CSchScheduleDictionary::CSchScheduleDictionary()
552 CSchScheduleIndex::CSchScheduleDictionary::~CSchScheduleDictionary()
557 void CSchScheduleIndex::CSchScheduleDictionary::ConstructL()
559 const TInt KGranularity = 3;
560 iMappings = new(ELeave) CArrayFixSeg<TSchScheduleMapplet>(KGranularity);
563 CSchScheduleIndex::CSchScheduleDictionary* CSchScheduleIndex::CSchScheduleDictionary::NewLC()
565 CSchScheduleDictionary* self = new(ELeave) CSchScheduleDictionary();
566 CleanupStack::PushL(self);
571 void CSchScheduleIndex::CSchScheduleDictionary::AssignL(TInt aKey, TStreamId aId)
573 if (aId == KNullStreamId)
575 Remove(aKey); // default associated stream is null
579 TSchScheduleMapplet entry(aKey, KNullStreamId);
580 TKeyArrayFix key(TSchScheduleMapplet::KeyOffset(), ECmpTInt32);
582 if (iMappings->FindIsq(entry, key, i) == 0)
584 iMappings->At(i).SetStream(aId);
588 entry.SetStream(aId);
589 iMappings->InsertIsqL(entry, key);
592 void CSchScheduleIndex::CSchScheduleDictionary::Remove(TInt aKey)
594 TSchScheduleMapplet entry(aKey, KNullStreamId);
595 TKeyArrayFix key(TSchScheduleMapplet::KeyOffset(), ECmpTInt32);
597 if (iMappings->FindIsq(entry, key, i) == 0)
598 iMappings->Delete(i);
601 TInt CSchScheduleIndex::CSchScheduleDictionary::Count() const
603 return iMappings->Count();
606 TStreamId CSchScheduleIndex::CSchScheduleDictionary::At(TInt aKey) const
608 TSchScheduleMapplet entry(aKey, KNullStreamId);
609 TKeyArrayFix key(TSchScheduleMapplet::KeyOffset(), ECmpTInt32);
611 if (iMappings->FindIsq(entry, key, i) != 0)
612 return KNullStreamId;
614 return iMappings->At(i).Stream();
617 TStreamId CSchScheduleIndex::CSchScheduleDictionary::AtIndex(TInt aIndex) const
619 return iMappings->At(aIndex).Stream();
622 void CSchScheduleIndex::CSchScheduleDictionary::InternalizeL(RReadStream& aStream)
624 aStream >> *iMappings;
627 void CSchScheduleIndex::CSchScheduleDictionary::ExternalizeL(RWriteStream& aStream) const
629 aStream << *iMappings;
633 // -----> CSchClientIndex (header)
636 CSchClientIndex::CSchClientIndex(RFs& aFsSession)
637 : iFsSession(aFsSession)
641 CSchClientIndex* CSchClientIndex::NewL(RFs& aFsSession)
643 return new(ELeave) CSchClientIndex(aFsSession);
646 void CSchClientIndex::AppendClientToListL(TPriQue<CClientProxy>& aClients, CClientProxy* aClient)
648 CClientProxy* currentClient;
649 TDblQueIter<CClientProxy> clientIter(aClients);
650 clientIter.SetToFirst();
651 while ((currentClient = clientIter++) != NULL)
653 //if match on same client name and priority
654 if (currentClient->IsEqual(aClient->ExecutorFileName(),aClient->Priority()))
656 //transfer all the tasks in aClient to this client
657 aClient->TransferTasks(*currentClient);
658 //now that we have transferred all the task ownership, we can safely delete the source client
663 //since that there is no matching one just add the aClient directly
664 aClients.Add(*aClient);
667 void CSchClientIndex::RestoreClientsL(TPriQue<CClientProxy>& aClients,
668 CStreamStore& aStore,
669 TStreamId aIndexStream,
670 CSchLogManager& aSchLogManager,TBool aBUR)
672 CSchClientDictionary* dictionary = DictionaryLC(aStore, aIndexStream);
674 // Restore every schedule in the dictionary
675 const TInt count = dictionary->Count();
676 LOGSTRING2("CSchClientIndex::RestoreClientsL - read %d dictionary entries", count);
678 for(TInt i=0; i<count; i++)
681 TStreamId streamId = dictionary->AtIndex(i);
683 RStoreReadStream stream;
684 stream.OpenLC(aStore, streamId);
687 CClientProxy* client = CClientProxy::NewL(iFsSession,stream,aSchLogManager);
688 CleanupStack::PopAndDestroy(); // stream
691 // only when this is called through restore we need to append to existing client which
692 // might already contain other transient tasks
695 AppendClientToListL(aClients,client);
699 aClients.Add(*client); // takes ownership
701 LOGSTRING2("CSchClientIndex::RestoreClientsL - restored client: %S", &client->ExecutorFileName());
703 CleanupStack::PopAndDestroy(dictionary);
706 TStreamId CSchClientIndex::CreateEmptyIndexL(CStreamStore& aStore) const
708 CSchClientDictionary* dictionary = CSchClientDictionary::NewLC();
709 RStoreWriteStream stream;
710 TStreamId id = stream.CreateLC(aStore);
711 stream << *dictionary;
713 CleanupStack::PopAndDestroy(2); // stream, dictionary
717 void CSchClientIndex::AddL(TStreamId aIndexStream, CStreamStore& aStore, const CClientProxy& aClient)
719 RStoreWriteStream stream;
720 TStreamId id = stream.CreateLC(aStore);
723 CleanupStack::PopAndDestroy(); // stream
725 // Read the dictionary and update an entry
726 CSchClientDictionary* dictionary = DictionaryLC(aStore, aIndexStream);
727 dictionary->AssignL(aClient.ExecutorFileName(), id);
729 // Store the dictionary
730 StoreDictionaryL(aStore, *dictionary, aIndexStream);
731 CleanupStack::PopAndDestroy(dictionary);
734 void CSchClientIndex::EditL(TStreamId aIndexStream, CStreamStore& aStore, const CClientProxy& aClient)
736 // Locate the existing stream (to be replaced)
737 CSchClientDictionary* dictionary = DictionaryLC(aStore, aIndexStream);
738 TStreamId id = dictionary->AtL(aClient.ExecutorFileName());
739 CleanupStack::PopAndDestroy(dictionary);
740 if (id == KNullStreamId)
742 // Wasn't found, add it instead...
743 AddL(aIndexStream, aStore, aClient);
747 // Replace - the original stream is orphaned but the new
748 // stream retains the old id.
749 RStoreWriteStream stream;
750 stream.ReplaceLC(aStore, id);
753 CleanupStack::PopAndDestroy(); // stream
757 void CSchClientIndex::DeleteL(TStreamId aIndexStream, CStreamStore& aStore, const CClientProxy& aClient)
759 // Locate the existing stream (to be deleted)
760 CSchClientDictionary* dictionary = DictionaryLC(aStore, aIndexStream);
761 TStreamId id = dictionary->AtL(aClient.ExecutorFileName());
762 __ASSERT_ALWAYS(id != KNullStreamId, SchBackupManagerUtils::Panic(SchBackupManagerUtils::ESchBackupManagerScheduleStreamToDeleteNotFound));
767 // Save changes to store - we have to do this here since
768 // we must assure that the store is fully updated before
769 // we remove this stream from the dictionary (in order to
770 // maintain integrity).
773 // Remove entry from the stream dictionary
774 dictionary->RemoveL(aClient.ExecutorFileName());
775 StoreDictionaryL(aStore, *dictionary, aIndexStream);
776 CleanupStack::PopAndDestroy(dictionary);
779 CSchClientIndex::CSchClientDictionary* CSchClientIndex::DictionaryLC(CStreamStore& aStore, TStreamId aIndexStream)
781 LOGSTRING("CSchClientIndex::DictionaryLC - restoring dictionary");
782 CSchClientDictionary* dictionary = CSchClientDictionary::NewLC();
783 RStoreReadStream stream;
784 stream.OpenLC(aStore, aIndexStream);
785 stream >> *dictionary;
786 CleanupStack::PopAndDestroy(); // stream
787 LOGSTRING("CSchClientIndex::DictionaryLC - dictionary restored");
791 void CSchClientIndex::StoreDictionaryL(CStreamStore& aStore, const CSchClientDictionary& aDictionary, TStreamId aStreamToReplace)
793 LOGSTRING("CSchClientIndex::DictionaryLC - storing dictionary");
794 RStoreWriteStream stream;
795 stream.ReplaceLC(aStore, aStreamToReplace);
796 stream << aDictionary;
798 CleanupStack::PopAndDestroy(); // stream
799 LOGSTRING("CSchClientIndex::DictionaryLC - dictionary stored");
803 // -----> TKeyArrayPtr (header)
805 NONSHARABLE_CLASS(TKeyArrayMapping) : public TKeyArrayFix
808 inline TKeyArrayMapping(TInt aOffset) : TKeyArrayFix(aOffset, TKeyCmpText()) {iHBufType=ETrue;}
810 virtual TAny* At(TInt aIndex) const;
811 virtual TInt Compare(TInt aLeft,TInt aRight) const;
817 TAny* TKeyArrayMapping::At(TInt aIndex) const
819 if (aIndex==KIndexPtr)
821 const CSchClientIndex::CSchClientMapplet** ppItem = (const CSchClientIndex::CSchClientMapplet**) iPtr;
822 const CSchClientIndex::CSchClientMapplet* pItem = *ppItem;
823 return (TAny*) pItem;
827 TInt baseOffset = aIndex * sizeof(const CSchClientIndex::CSchClientMapplet*);
828 const TUint8* pItemUncast = iBase->Ptr(baseOffset).Ptr();
829 const CSchClientIndex::CSchClientMapplet** ppItem = (const CSchClientIndex::CSchClientMapplet**) pItemUncast;
830 const CSchClientIndex::CSchClientMapplet* pItem = *ppItem;
831 return (TAny*) pItem;
835 TInt TKeyArrayMapping::Compare(TInt aLeft,TInt aRight) const
839 const CSchClientIndex::CSchClientMapplet* left = (const CSchClientIndex::CSchClientMapplet*) At(aLeft);
840 const CSchClientIndex::CSchClientMapplet* right = (const CSchClientIndex::CSchClientMapplet*) At(aRight);
842 return (left->Key().Compare(right->Key()));
850 // -----> CSchClientIndex::CSchClientMapplet (header)
853 CSchClientIndex::CSchClientMapplet::~CSchClientMapplet()
858 CSchClientIndex::CSchClientMapplet* CSchClientIndex::CSchClientMapplet::NewLC(RReadStream& aStream)
860 CSchClientMapplet* self = new(ELeave) CSchClientMapplet;
861 CleanupStack::PushL(self);
866 CSchClientIndex::CSchClientMapplet* CSchClientIndex::CSchClientMapplet::NewLC(const TDesC& aKey, TStreamId aStream)
868 CSchClientMapplet* self = new(ELeave) CSchClientMapplet;
869 CleanupStack::PushL(self);
870 self->iKey = aKey.AllocL();
871 self->iStream = aStream;
875 void CSchClientIndex::CSchClientMapplet::InternalizeL(RReadStream& aStream)
877 HBufC* key = HBufC::NewLC(aStream, KMaxTInt);
880 CleanupStack::Pop(); // key
884 void CSchClientIndex::CSchClientMapplet::ExternalizeL(RWriteStream& aStream) const
892 // -----> CSchClientIndex::CSchClientDictionary (header)
895 CSchClientIndex::CSchClientDictionary::CSchClientDictionary()
899 CSchClientIndex::CSchClientDictionary::~CSchClientDictionary()
903 for(int index = 0; index < iMappings->Count();index++)
905 delete iMappings->At(index);
911 void CSchClientIndex::CSchClientDictionary::ConstructL()
913 const TInt KGranularity = 2;
914 iMappings = new(ELeave) CArrayPtrSeg<CSchClientMapplet>(KGranularity);
917 CSchClientIndex::CSchClientDictionary* CSchClientIndex::CSchClientDictionary::NewLC()
919 CSchClientDictionary* self = new(ELeave) CSchClientDictionary();
920 CleanupStack::PushL(self);
925 void CSchClientIndex::CSchClientDictionary::AssignL(const TDesC& aKey, TStreamId aId)
927 if (aId == KNullStreamId)
929 RemoveL(aKey); // default associated stream is null
933 CSchClientMapplet* entry = CSchClientMapplet::NewLC(aKey, KNullStreamId);
934 TKeyArrayMapping key(CSchClientMapplet::KeyOffset());
936 if (iMappings->FindIsq(entry, key, i) == 0)
938 iMappings->At(i)->SetStream(aId);
939 CleanupStack::PopAndDestroy(); // entry
943 entry->SetStream(aId);
944 iMappings->InsertIsqL(entry, key);
945 CleanupStack::Pop(); // entry
948 void CSchClientIndex::CSchClientDictionary::RemoveL(const TDesC& aKey)
950 CSchClientMapplet* entry = CSchClientMapplet::NewLC(aKey, KNullStreamId);
951 TKeyArrayMapping key(CSchClientMapplet::KeyOffset());
953 if (iMappings->FindIsq(entry, key, i) == 0)
955 delete iMappings->At(i);
956 iMappings->Delete(i);
958 CleanupStack::PopAndDestroy(); // entry
961 TInt CSchClientIndex::CSchClientDictionary::Count() const
963 return iMappings->Count();
966 TStreamId CSchClientIndex::CSchClientDictionary::AtL(const TDesC& aKey) const
968 CSchClientMapplet* entry = CSchClientMapplet::NewLC(aKey, KNullStreamId);
969 TKeyArrayMapping key(CSchClientMapplet::KeyOffset());
971 TInt found = iMappings->FindIsq(entry, key, i);
972 CleanupStack::PopAndDestroy(); // entry
974 return KNullStreamId;
975 return iMappings->At(i)->Stream();
978 TStreamId CSchClientIndex::CSchClientDictionary::AtIndex(TInt aIndex) const
980 return iMappings->At(aIndex)->Stream();
983 void CSchClientIndex::CSchClientDictionary::InternalizeL(RReadStream& aStream)
985 const TInt count = aStream.ReadInt32L();
986 for(TInt i=0; i<count; i++)
988 CSchClientMapplet* mapplet = CSchClientMapplet::NewLC(aStream);
989 iMappings->AppendL(mapplet);
990 CleanupStack::Pop(); // mapplet
994 void CSchClientIndex::CSchClientDictionary::ExternalizeL(RWriteStream& aStream) const
996 const TInt count = iMappings->Count();
997 aStream.WriteInt32L(count);
998 for(TInt i=0; i<count; i++)
999 aStream << *iMappings->At(i);