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.
19 #include "srvrepos_noc.h"
20 #include "obsrvr_noc.h"
21 #include "srvparams.h"
23 CSharedRepository* CSharedRepository::NewL(TUid aUid)
25 CSharedRepository* self = new(ELeave) CSharedRepository();
26 CleanupStack::PushL(self);
27 self->ConstructL(aUid);
28 CleanupStack::Pop(self);
30 // debug check that CRepository::TTransactionMode modes match those used internally
31 // from CRepositoryTransactor: internal state logic relies on this
32 // there should be a better location for these checks...
33 ASSERT(CRepository::EReadTransaction == static_cast<CRepository::TTransactionMode>(EReadTransaction));
34 ASSERT(CRepository::EConcurrentReadWriteTransaction == static_cast<CRepository::TTransactionMode>(EConcurrentReadWriteTransaction));
35 ASSERT(CRepository::EReadWriteTransaction == static_cast<CRepository::TTransactionMode>(EReadWriteTransaction));
40 void CSharedRepository::ConstructL(TUid aUid)
42 iSimRep = CHeapRepository::NewL(aUid);
45 CSharedRepository::CSharedRepository() : iNotificationState(ETrue)
49 CSharedRepository::~CSharedRepository()
57 TUid CSharedRepository::Uid() const
59 return iSimRep->Uid();
63 Stores the repository in-memory content to the related repository file on drive C.
64 If the operation fails, the in-memory content won't match the content of
65 the repository file (which will be kept as it was before the CommitChangesL() call).
66 In order to keep the consistency, the in-memory repository content is deleted now
67 and restored later, on the next repository operation.
69 TInt CSharedRepository::CommitChanges(TCentRepLocation aLocation)
71 iInconsistentData=ETrue;
73 HBufC* filePath(NULL);
74 TRAPD(err,TServerResources::CreateRepositoryFileNameL(filePath,iSimRep->Uid(),aLocation,ECre));
77 iSimRep->ResetContent();
81 // should not be committing while transactions are still active
82 ASSERT(!IsTransactionActive());
83 TInt ret=iSimRep->CommitChanges(TServerResources::iFs,TServerResources::iPersistsVersion,*filePath);
86 iInconsistentData=EFalse;
92 // merge transaction settings (which may include entries flagged as deleted), persist and notify
93 // private method relies on calling code to ensure it is permitted to make changes here.
94 // if this method is committing any changes, it cancels all other sessions' transactions
95 TInt CSharedRepository::DoCommitTransactionSettings(CRepositoryTransactor& aTransactor, TUint32& aKeyInfo)
97 aKeyInfo = KUnspecifiedKey;
98 if (0 == aTransactor.iTransactionSettings.Count())
100 aKeyInfo = 0; // == number of settings modified
101 return KErrNone; // nothing to do
103 TInt error = iSimRep->SettingsArray().MergeArray(aTransactor.iTransactionSettings, iSimRep->DeletedSettingsArray(), ETransactionMerge);
104 TInt numChanges = aTransactor.iTransactionSettings.Count();
107 if (error == KErrNone)
109 aKeyInfo = 0; // no changes
111 // no changes were made, so the internal cache is still valid.
112 // This could be because there were no changes: empty list, only deletes on
113 // non-existent items (a setting created and deleted in the transaction),
114 // or because of error, such as failure of an initial realloc failure.
117 if (error != KErrNone)
119 // the repository is corrupted. Dump it for lazy loading later
122 // mark cache as inconsistent so it is reloaded.
123 iInconsistentData = ETrue;
126 if (error == KErrNone)
128 // changes have been made: fail all other sessions' transactions so we can commit
129 FailAllTransactions(/*aExcludeTransactor=*/&aTransactor);
130 error = CommitChanges(); // this already calls ResetContent() in case of failure
132 if (error == KErrNone)
134 // settings are now persistent on disk: we can now notify about the changes
135 // following will notify about objects that are created and deleted in the transaction
136 // this could be made faster by having a multiple Notify method.
137 // That would also allow Notify messages to be more descriptive - ranges of Keys
138 for (TInt i = 0; i < numChanges; i++)
140 Notify(aTransactor.iTransactionSettings[i].Key());
142 aKeyInfo = /*reinterpret_cast<TUint32>*/numChanges;
147 void CSharedRepository::SetMetaDataOnRead(TServerSetting& aSetting, TBool aSingleMetaFound)
149 iSimRep->SetMetaDataOnRead(aSetting, aSingleMetaFound);
152 void CSharedRepository::SetMetaDataOnCreate(TServerSetting& aNewSetting, TUint32* aMeta)
156 aNewSetting.SetMeta(*aMeta);
160 // No metadata specified. First check for a matching "range" default
162 TSettingsDefaultMeta* defaultMeta = iSimRep->RangeMetaArray().Find(aNewSetting.Key());
165 aNewSetting.SetMeta(defaultMeta->GetDefaultMetadata());
169 // Range value not found, try for a repository default
170 aNewSetting.SetMeta(iSimRep->DefaultMeta());
175 void CSharedRepository::CreateL(TServerSetting& aSetting, TSettingsAccessPolicy*& aPolicy, TBool aFirstLoad, TBool aSingleMetaFound)
177 User::LeaveIfError(iSimRep->Create(aSetting, aPolicy, aSingleMetaFound));
180 Notify(aSetting.Key());
184 // deletes an individual setting in the shared repository and makes it persistent
185 // if changes are made, all sessions' transactions are failed
186 TInt CSharedRepository::DeleteAndPersist(TUint32 aId)
188 TServerSetting* s = iSimRep->SettingsArray().Find(aId);
191 iSimRep->SettingsArray().Remove(aId);
193 // removed a setting, so must fail all sessions' transactions before commit possible
194 FailAllTransactions(/*aExcludeTransactor=*/NULL);
195 TInt error = CommitChanges();
196 if (error == KErrNone)
203 // deletes an individual setting without making it persistent
204 // must not be called while any sessions are in transactions
205 TInt CSharedRepository::DeleteNoPersist(TUint32 aId)
207 // should only be calling this if no transactions are active
208 ASSERT(!IsTransactionActive());
209 TServerSetting* s = iSimRep->SettingsArray().Find(aId);
213 iSimRep->SettingsArray().Remove(aId);
217 TInt CSharedRepository::ResetNoPersistL(TServerSetting& aSetting)
219 TServerSetting* s = iSimRep->SettingsArray().Find(aSetting.Key());
220 if ((!s) || (*s != aSetting))
224 // save access policy of setting
225 TSettingsAccessPolicy* policy=s->AccessPolicy();
226 s->Transfer(aSetting);
227 // restore access policy of setting
228 s->SetAccessPolicy(policy);
232 TServerSetting setting;
233 setting.Transfer(aSetting);
234 setting.SetAccessPolicy(GetFallbackAccessPolicy(setting.Key()));
236 iSimRep->SettingsArray().OrderedInsertL(setting);
238 TInt index = iSimRep->DeletedSettingsArray().FindInUnsignedKeyOrder(aSetting.Key());
239 if (index != KErrNotFound)
240 iSimRep->DeletedSettingsArray().Remove(index);
252 // if changes are made, all sessions' transactions are failed
253 void CSharedRepository::ResetAndPersistL(TServerSetting& aSetting)
255 if (ResetNoPersistL(aSetting) == KErrNone)
257 // changed a setting, so must fail all sessions' transactions
258 // before commit possible
259 FailAllTransactions(/*aExcludeTransactor=*/NULL);
261 Notify(aSetting.Key());
265 TInt CSharedRepository::ResetAllNoPersistL(CSharedRepository& aNewContent)
267 // mark cache as inconsistent in case Reset fails, so it is reloaded.
268 iInconsistentData=ETrue;
270 // should not change repository while transactions in progress: should fail them first
271 ASSERT(!IsTransactionActive());
272 TInt newCount = (aNewContent.iSimRep)->SettingsArray().Count();
273 TInt count = iSimRep->SettingsArray().Count();
278 while(newIndex<newCount && index<count)
280 const TServerSetting& newSetting = (aNewContent.iSimRep)->SettingsArray()[newIndex];
281 const TServerSetting& setting = iSimRep->SettingsArray()[index];
283 TUint32 newKey = newSetting.Key();
284 TUint32 key = setting.Key();
293 if(newSetting!=setting)
307 while(newIndex<newCount)
309 Notify((aNewContent.iSimRep)->SettingsArray()[newIndex++].Key());
314 Notify(iSimRep->SettingsArray()[index++].Key());
317 // Replace current settings with settings read from ROM, this
318 // will leave settings pointing to new single policies
319 iSimRep->SettingsArray().AdoptL((aNewContent.iSimRep)->SettingsArray());
321 // Reset policy pointers to point at this repositories policies
322 newCount=iSimRep->SettingsArray().Count();
323 for(TInt i=0; i<newCount;i++)
325 (iSimRep->SettingsArray())[i].SetAccessPolicy(NULL);
326 TUint32 key = (iSimRep->SettingsArray())[i].Key();
327 (iSimRep->SettingsArray())[i].SetAccessPolicy(GetFallbackAccessPolicy(key));
330 iSimRep->DeletedSettingsArray().Reset();
332 iInconsistentData=EFalse;
336 // returns the read security policy used if there is no per-setting policy at aId
337 const TSecurityPolicy& CSharedRepository::GetFallbackReadAccessPolicy(TUint32 aId)
339 return iSimRep->GetFallbackReadAccessPolicy(aId);
342 // returns the write security policy used if there is no per-setting policy at aId
343 const TSecurityPolicy& CSharedRepository::GetFallbackWriteAccessPolicy(TUint32 aId)
345 return iSimRep->GetFallbackWriteAccessPolicy(aId);
348 // Get pointer to security policy that applies to a given setting
349 TSettingsAccessPolicy* CSharedRepository::GetFallbackAccessPolicy(TUint32 aId
350 #ifdef SYMBIAN_CENTREP_SUPPORT_MULTIROFS
355 return iSimRep->GetFallbackAccessPolicy(aId
356 #ifdef SYMBIAN_CENTREP_SUPPORT_MULTIROFS
363 TInt CSharedRepository::ReadSettingSavePolicyL(CIniFileIn& aFile,TServerSetting& aSetting, TSettingsAccessPolicy*& aPolicy, TBool& aSingleMetaFound)
365 return iSimRep->ReadSettingSavePolicyL(aFile, aSetting, aPolicy, aSingleMetaFound);
368 // Merge settings in this->iSettings with the iSettings of aMergeRep
369 // During an intsall/upgrade event aMergeRep will be created from the installed file
370 // During an upinstall event aMergeRep will be created from the ROM file
371 void CSharedRepository::MergeL(CSharedRepository& aMergeRep, TMergeType aMergeType)
373 // Process settings from main section - this updates values only
374 User::LeaveIfError(GetSettings().MergeArray(aMergeRep.GetSettings(), iSimRep->DeletedSettingsArray(), aMergeType));
376 //if the merging is due to a ROM Flash, we need to copy over both the NEW ROM keypsace global properties
377 //(default access policies/metadata and range policies/metadata), individual policies, we then need to ensure
378 //that the settings point at the correct individual policies and metadata.
379 if (aMergeType==ERomFlash)
381 //copy the default/range/individual policy
382 iSimRep->SetDefaultPolicy(aMergeRep.iSimRep->GetDefaultAccessPolicy());
383 #ifdef SYMBIAN_CENTREP_SUPPORT_MULTIROFS
384 iSimRep->GetDefaultAccessPolicy().iHighKey=aMergeRep.iSimRep->GetDefaultAccessPolicy().iHighKey;
385 iSimRep->GetDefaultAccessPolicy().iKeyMask=aMergeRep.iSimRep->GetDefaultAccessPolicy().iKeyMask;
387 iSimRep->RangePolicyArray().Reset();
388 TInt count=aMergeRep.iSimRep->RangePolicyArray().Count();
389 iSimRep->RangePolicyArray().ReserveL(count);
390 for (TInt i=0;i<count;i++)
392 iSimRep->RangePolicyArray().AppendL(aMergeRep.iSimRep->RangePolicyArray()[i]);
394 iSimRep->SinglePolicyArray().ResetAndDestroy();
395 count=aMergeRep.iSimRep->SinglePolicyArray().Count();
396 iSimRep->SinglePolicyArray().ReserveL(count);
397 for (TInt i=0;i<count;i++)
399 iSimRep->SinglePolicyArray().AppendL(aMergeRep.iSimRep->SinglePolicyArray()[i]);
401 //now need to reset the aMergeRep single policies so it is not going to destroy the
402 //individual policies as ownership has been transferred
403 aMergeRep.iSimRep->SinglePolicyArray().Reset();
405 //copy the default/range metadata
406 iSimRep->SetDefaultMeta(aMergeRep.iSimRep->DefaultMeta());
407 iSimRep->RangeMetaArray().Reset();
408 count=aMergeRep.iSimRep->RangeMetaArray().Count();
409 iSimRep->RangeMetaArray().ReserveL(count);
410 for (TInt i=0;i<count;i++)
412 iSimRep->RangeMetaArray().AppendL(aMergeRep.iSimRep->RangeMetaArray()[i]);
415 //set the timestamp,owner etc
416 iSimRep->SetTimeStamp(aMergeRep.iSimRep->TimeStamp());
417 iSimRep->SetOwner(aMergeRep.iSimRep->Owner());
420 // Update all access policies and meta
421 for(TInt i=0; i<iSimRep->SettingsArray().Count();i++)
423 TServerSetting& setting= iSimRep->SettingsArray()[i];
424 setting.SetAccessPolicy(GetFallbackAccessPolicy(setting.Key()));
428 // Save timestamp of installed file
429 void CSharedRepository::SetInstallTime(TTime aInstallTime)
431 iSimRep->SetTimeStamp(aInstallTime);
434 // Handle creation or upgrade of file in install directory
435 void CSharedRepository::HandleUpdateMergeL(TTime aInstallFileTimeStamp, CSharedRepository& aInstallRep)
437 MergeL(aInstallRep, ESWIUpgradeMerge);
439 SetInstallTime(aInstallFileTimeStamp); // Set merge timestamp
440 CommitChangesL(); // Commit changes to write system drive file
442 // settings are now persistent on disk: we can now notify about the changes
443 for (TInt i = 0; i < (aInstallRep.iSimRep)->SettingsArray().Count(); i++)
445 Notify((aInstallRep.iSimRep)->SettingsArray()[i].Key());
449 // Handle merge activity due to an uninstall
450 void CSharedRepository::HandleDeleteMergeL(CSharedRepository& aRomRep)
452 MergeL(aRomRep, ESWIDowngradeMerge);
454 SetInstallTime(0); // Reset timestamp
455 CommitChangesL(); // Commit changes to write system drive file
457 // settings are now persistent on disk: we can now notify about the changes
458 for (TInt i = 0; i < (aRomRep.iSimRep)->SettingsArray().Count(); i++)
460 Notify((aRomRep.iSimRep)->SettingsArray()[i].Key());
465 #ifdef CENTREP_CONV_TOOL
467 Statement "iInconsistentData = ETrue;" must be the first statement in the method,
468 "iInconsistentData = EFalse;" must be the last. It is used for lasy-load implementation
469 for the repository and solves the problem that if CommitChangesL() fails the in-memory
470 repository data won't match the repository data, stored in the file.
471 This routine is being retained for testing purposes
473 void CSharedRepository::DoCommitChangesToIniFileL(const TDesC& aOutFileName
474 #ifdef SYMBIAN_CENTREP_SUPPORT_MULTIROFS
479 iInconsistentData=ETrue;
481 // should not be committing while transactions are still active
482 ASSERT(!IsTransactionActive());
484 iSimRep->DoCommitChangesToIniFileL(TServerResources::iFs,aOutFileName
485 #ifdef SYMBIAN_CENTREP_SUPPORT_MULTIROFS
490 iInconsistentData = EFalse;
492 #endif //CENTREP_CONV_TOOL
495 The method reloads the repository content from a repository file.
496 The current repository must be emptied (or must be empty already) before the call is made.
497 @param aIniFile A reference to CIniFileIn object, which will be used to load
498 the repository content.
499 @param aFirstLoad is used to indicate whether the file is reloaded for first time, this is used to prevent
500 notification if not needed. For example file loading for merging purpose will not result in notification
501 @return KErrCorrupt Corrupted repository file.
502 KErrNone The repository content was seccessfully loaded into memory.
503 KErrNotFound Setting not found in the file.
504 @leave System-wide error codes.
505 @leave KErrGeneral It's probably a programmer's error - current CSharedRepository
506 object is partially initialised.
508 TInt CSharedRepository::ReloadContentL(CIniFileIn& aIniFile, TBool aFirstLoad)
510 // Preconditions - CHeapRepository object should be an empty one.
511 if(!iSimRep->IsEmpty())
513 User::Leave(KErrGeneral);
515 TInt err = iSimRep->ReloadContentExceptSettingsL(aIniFile);
516 if(err == KErrCorrupt)
520 CleanupClosePushL(iSimRep->RangeMetaArray());
521 CleanupClosePushL(iSimRep->RangePolicyArray());
524 TServerSetting setting;
525 TSettingsAccessPolicy* policy;
526 TBool singleMetaFound;
527 TCleanupItem tc(CHeapRepository::SinglePoliciesCleanup, &(iSimRep->SinglePolicyArray()));
528 CleanupStack::PushL(tc);
529 CleanupClosePushL(iSimRep->SettingsArray());
530 while((err = ReadSettingSavePolicyL(aIniFile, setting, policy, singleMetaFound)) == KErrNone)
533 if(iSimRep->SettingsArray().IsDefault())
537 CreateL(setting, policy, aFirstLoad, singleMetaFound);
540 if(err == KErrNotFound)
546 CleanupStack::Pop(4,&(iSimRep->RangeMetaArray()));
550 CleanupStack::PopAndDestroy(4,&(iSimRep->RangeMetaArray()));
556 Resets current repository data - actually all of them, which may be loaded from
557 the related ini file.
558 The iUid data member value is kept as it was at the moment of creation of
559 CSharedRepository object.
561 void CSharedRepository::ResetContent()
563 iSimRep->ResetContent();
567 This function is used to restore the notification, which was temporary disabled
568 when making RestoreConsistencyL() call.
569 @param aNotificationState It points to CObservable::iNotificationState data member, which
570 controls the notification state - active or disabled.
573 static void RestoreNotification(void* aNotificationState)
575 TBool* notificationState = static_cast <TBool*> (aNotificationState);
576 *notificationState = ETrue;
580 The method reloads the repository content from the related ini file if previous
581 CommitChangesL() has not completed successfully.
583 void CSharedRepository::RestoreConsistencyL()
585 //Do nothing if previous CommitChangesL() completed successfully.
586 if (!iInconsistentData)
590 //Reset current repository data
592 //Disable notifications
593 TCleanupItem restoreNotification(&RestoreNotification, &iNotificationState);
594 CleanupStack::PushL(restoreNotification);
595 iNotificationState = EFalse;
596 //Reload the repository content from the related ini file
597 DoRestoreConsistencyL();
598 //Activate notifications
599 CleanupStack::PopAndDestroy();//restoreNotification
601 TCentRepLocation location = EPersists;
602 HBufC* persistsTmpFilePath(NULL);
603 //allocates memory on the heap
604 TServerResources::CreateRepositoryFileNameLC(persistsTmpFilePath,iSimRep->Uid(),location,ETmp);
605 // Remove any .tmp file
606 // If a debug build - record error
607 TInt fileDeleteErr=TServerResources::iFs.Delete(*persistsTmpFilePath);
608 if ((fileDeleteErr != KErrNone) && (fileDeleteErr != KErrNotFound))
611 RDebug::Print(_L("CHeapRepository::RestoreConsistencyL - Failed to delete file. Error = %d"), fileDeleteErr);
615 CleanupStack::PopAndDestroy(persistsTmpFilePath);
617 iInconsistentData=EFalse;
621 The method reloads the repository content from the related cre or ini file.
622 @leave System-wide error codes
624 void CSharedRepository::DoRestoreConsistencyL()
626 #ifdef SYMBIAN_CENTREP_SUPPORT_MULTIROFS
627 //note that the function below already handles the deletion of any corrupt file
628 //in non-rom location
629 TInt ret=TServerResources::iObserver->CreateRepositoryL(this,EPersists);
630 if (ret==KErrNotFound)
632 ret=TServerResources::iObserver->CreateRepositoryL(this,EInstall);
633 if (ret==KErrNotFound)
635 ret=TServerResources::iObserver->CreateRepositoryL(this,ERom);
636 User::LeaveIfError(ret);
640 TCentRepLocation location;
642 TInt err = FindLocationForFileL(location,iSimRep->Uid(),ECre);
643 if (err != KErrNotFound)
645 User::LeaveIfError(CreateRepositoryFromCreFileL(location));
649 User::LeaveIfError(FindLocationForFileL(location,iSimRep->Uid(),EIni));
651 HBufC* fileName(NULL);
652 TServerResources::CreateRepositoryFileNameLC(fileName,iSimRep->Uid(),location,EIni);
654 CIniFileIn* iniFile = NULL;
655 err = CIniFileIn::NewLC(TServerResources::iFs,iniFile,*fileName);
656 if (err==KErrCorrupt && location!=ERom)
658 User::LeaveIfError(TServerResources::iFs.Delete(*fileName));
660 User::LeaveIfError(err);
663 err = ReloadContentL(*iniFile);
664 User::LeaveIfError(err);
666 CleanupStack::PopAndDestroy(iniFile); //iniFile
667 CleanupStack::PopAndDestroy(fileName); //fileName
673 This method looks for and sets a location for a given repository.
674 It is based on EAuto mode thus it goes through all locations in the
675 same order (EPersists - EInstall - ERom)
677 @param aLocation - returns a location for a repository
678 @param aUid - id of a repository which location should be found
679 @param aType - repository file type (.txt or .cre)
680 @return KErrNone if aLocation succesfully set for a given repository,
681 KErrNotFound if a repository was not found in any locations.
685 #ifndef SYMBIAN_CENTREP_SUPPORT_MULTIROFS
686 TInt CSharedRepository::FindLocationForFileL(TCentRepLocation& aLocation,TUid aUid,const TCentRepFileType aType) const
688 if(TServerResources::CentrepFileExistsL(aUid, EPersists, aType))
690 aLocation = EPersists;
694 if(TServerResources::CentrepFileExistsL(aUid, EInstall, aType))
696 aLocation = EInstall;
700 if(TServerResources::CentrepFileExistsL(aUid, ERom, aType))
710 TInt CSharedRepository::CreateRepositoryFromCreFileL( TCentRepLocation aLocation)
712 // Get file path name from location
713 HBufC* filePath(NULL);
714 TServerResources::CreateRepositoryFileNameLC(filePath,iSimRep->Uid(), aLocation,ECre);
715 // Trap errors from repository creation so we can delete corrupt repositories
716 TRAPD(error, iSimRep->CreateRepositoryFromCreFileL(TServerResources::iFs,*filePath));
717 if(error!=KErrNone && error!=KErrNotFound && error!=KErrNoMemory)
720 // store wasn't quite what we were expecting - can't return an error, can't leave
721 // so all we can do is close the file, tidy up as best we can, and return corrupt
722 if (aLocation != ERom)
724 // If a debug build - record error
725 TInt fileDeleteErr=TServerResources::iFs.Delete(*filePath);
726 if (fileDeleteErr != KErrNone)
729 RDebug::Print(_L("CSharedRepository::CreateRepositoryFromCreFileL - Failed to delete file. Error = %d"), fileDeleteErr);
735 else if( error==KErrNoMemory)
737 User::Leave(KErrNoMemory);
739 CleanupStack::PopAndDestroy(filePath);
743 /** Attempts to start a transaction.
744 Guaranteed to succeed (return KErrNone) for EConcurrentReadWriteTransaction mode only.
745 @param aTransactor transactor attempting to start transaction
746 @param aMode type of transaction to be started
747 @pre transactor is not in a transaction
748 @return KErrNone if the transaction is started, KErrLocked if read/write locks prevented that
749 type of transaction from starting now, and KErrArgument for invalid aMode.
750 @post On returning KErrNone, transaction is started and read/write locks are obtained for it
751 in the shared repository. Any other return: transaction has not started.
753 TInt CSharedRepository::StartTransaction(CRepositoryTransactor& aTransactor, TInt aMode)
755 // session can only be in one transaction
756 ASSERT(!aTransactor.IsInTransaction());
758 CObservable::TSharedRepositoryInfo* shrepinfo = TServerResources::iObserver->SharedRepositoryInfo(iSimRep->Uid());
762 case EConcurrentReadWriteTransaction:
763 // can always start this type of transaction
764 shrepinfo->iNumActiveConcurrentReadWriteTransactions++;
766 case EReadTransaction:
767 // negative lock means there is an active EReadWriteTransaction
768 if (shrepinfo->iPessimisticTransactionLockCount < 0)
770 ASSERT(shrepinfo->iPessimisticTransactionLockCount == -1); // sanity check
773 // when non-negative lock equals number of active EReadTransactions.
774 shrepinfo->iPessimisticTransactionLockCount++;
776 case EReadWriteTransaction:
777 // lock is zero if there are no active pessimistic transactions
778 if (shrepinfo->iPessimisticTransactionLockCount != 0)
782 // lock value of -1 means the exclusive EReadWriteTransaction is active
783 shrepinfo->iPessimisticTransactionLockCount = -1;
786 // not a valid transaction mode
789 aTransactor.AddToQueue(shrepinfo->iTransactors, aMode);
793 /** Commit transaction
794 @return KErrNone on success, or error code.
796 on success (return KErrNone): aKeyInfo returns number of modified settings;
797 on failure (other error code): KUnspecifiedKey
798 @pre transactor is in a transaction.
799 @post transactor is not in a transaction
801 TInt CSharedRepository::CommitTransaction(CRepositoryTransactor& aTransactor, TUint32& aKeyInfo)
803 // calling code should have panicked the client if not in a transaction
804 ASSERT(aTransactor.IsInTransaction());
805 TInt result = aTransactor.iTransactionResult;
806 if (aTransactor.IsInFailedTransaction())
808 ASSERT(result != KErrNone);
809 aKeyInfo = aTransactor.iTransactionErrorKey;
813 ASSERT(result == KErrNone);
814 ASSERT(aTransactor.iTransactionErrorKey == KUnspecifiedKey);
816 // must release locks otherwise shared repository will not commit changes
817 // failed transactions have already released their locks
818 TServerResources::iObserver->ReleaseTransactionLock(aTransactor,iSimRep->Uid());
821 // transactions that haven't made any changes can be closed at any time
822 if (aTransactor.IsInActiveReadWriteTransaction() &&
823 (aTransactor.iTransactionSettings.Count() > 0))
825 result = DoCommitTransactionSettings(aTransactor, aKeyInfo);
828 // transaction is complete - remove from queue
829 CObservable::TSharedRepositoryInfo* shrepinfo = TServerResources::iObserver->SharedRepositoryInfo(iSimRep->Uid());
831 shrepinfo->iTransactors.Remove(aTransactor);
832 //Remove the link to the next transaction
833 aTransactor.iLink.iNext = NULL;
839 TInt CSharedRepository::FailTransaction(CRepositoryTransactor& aTransactor, TInt aError, TUint32 aErrorKey)
841 ASSERT(aError != KErrNone); // must fail for a reason
842 if (aTransactor.IsInActiveTransaction())
844 // locks cannot be removed from a failed transaction, so release before failing
845 TServerResources::iObserver->ReleaseTransactionLock(aTransactor,iSimRep->Uid());
846 aTransactor.SetFailed(aError, aErrorKey);
848 return aError; // to allow "return FailTransaction(error, errorKey);" - error written once
851 /** Fails all active transactions - except for the optional aExcludeTransactor, releasing locks.
852 All transactions are failed with reason "KErrLocked" meaning they are "locked out".
853 This should only be done to allow another agent to change values in the repository.
854 Beware that all concurrent read/write transactions that are failed with KErrLocked are
855 expected to retry the transactions straight afterwards - must be careful to allow their
856 retry strategy to be successful.
858 void CSharedRepository::FailAllTransactions(const CRepositoryTransactor* aExcludeTransactor)
860 CObservable::TSharedRepositoryInfo* shrepinfo = TServerResources::iObserver->SharedRepositoryInfo(iSimRep->Uid());
862 TSglQueIter<CRepositoryTransactor> transIter(shrepinfo->iTransactors);
863 CRepositoryTransactor* transactor;
864 while ((transactor = transIter++) != NULL)
866 if (transactor != aExcludeTransactor)
868 FailTransaction(*transactor, KErrLocked, KUnspecifiedKey);
873 /** must currently be in active Read transaction. Does not fail
874 transaction here if promotion to read/write failed.
875 @return KErrNone if promoted, KErrLocked if not
877 TInt CSharedRepository::AttemptPromoteTransactionToReadWrite(CRepositoryTransactor& aTransactor)
879 // transactor should currently be in an active read transaction
880 ASSERT(aTransactor.IsInActiveReadTransaction());
882 CObservable::TSharedRepositoryInfo* shrepinfo = TServerResources::iObserver->SharedRepositoryInfo(iSimRep->Uid());
884 // sanity check: must only be pessimistic reads active
885 ASSERT(shrepinfo->iPessimisticTransactionLockCount > 0);
886 // can promote only if there are no other active read transactions:
887 if (1 == shrepinfo->iPessimisticTransactionLockCount)
889 // may only promote to exclusive read/write as it has the same commit semantics
890 // as Read transaction: concurrent R/W must wait for reads to finish first.
891 aTransactor.PromoteToExclusiveReadWrite();
892 // lock value of -1 means the exclusive EReadWriteTransaction is active
893 shrepinfo->iPessimisticTransactionLockCount = -1;
901 void CSharedRepository::ExternalizeCre(RWriteStream& aStream) const
903 iSimRep->ExternalizeCre(TServerResources::iPersistsVersion,aStream);
906 void CSharedRepository::InternalizeCreL(RReadStream& aStream)
908 iSimRep->InternalizeCreL(aStream);
911 #ifdef SYMBIAN_CENTREP_SUPPORT_MULTIROFS
912 void CSharedRepository::InternalizeCreL(RReadStream& aStream,TUint8& aCreVersion)
914 iSimRep->InternalizeCreL(aStream,aCreVersion);
918 void CSharedRepository::Notify(TUint32 aVal) const
920 if(iNotificationState)
922 TServerResources::iObserver->Notify(iSimRep->Uid(), aVal);
926 TBool CSharedRepository::IsTransactionActive()
928 CObservable::TSharedRepositoryInfo* shrepinfo = TServerResources::iObserver->SharedRepositoryInfo(iSimRep->Uid());
931 return (shrepinfo->iPessimisticTransactionLockCount != 0) ||
932 (shrepinfo->iNumActiveConcurrentReadWriteTransactions > 0);
937 RSettingsArray& CSharedRepository::GetSettings()
939 return iSimRep->SettingsArray();