os/persistentdata/persistentstorage/centralrepository/cenrepsrv/backup.cpp
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
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".
     7 //
     8 // Initial Contributors:
     9 // Nokia Corporation - initial contribution.
    10 //
    11 // Contributors:
    12 //
    13 // Description:
    14 //
    15 
    16 #include <e32def.h>
    17 #include <e32property.h>
    18 #include <s32file.h>
    19 #include <connect/sbdefs.h>
    20 #include "srvres.h"
    21 #include "cachemgr.h"
    22 #include "srvparams.h"
    23 #include "shrepos.h"
    24 #include "sessnotf.h"
    25 #include "backup.h"
    26 
    27 #define UNUSED_VAR(a) a = a
    28 
    29 _LIT (KBackupFileName, "BACKUP") ;
    30 _LIT (KRestoreFileName, "RESTORE") ;
    31 _LIT(KBackupFileExt, ".bak");
    32 _LIT(KRestoreFileExt, ".rst");
    33 
    34 
    35 TRepositoryBackupState CRepositoryBackupClient::iBackupStatus ;
    36 
    37 
    38 //
    39 // Backup stream index class - Used to hold association between a UID (in
    40 // our case the UID of a repository) and a stream ID - Can't use CStreamDictionary
    41 // because that only lets you retrieve stream IDs by a (previously known) UID rather
    42 // than iterate through its contentsretrieving UID/StreamID pairs...
    43 // 
    44 
    45 
    46 //
    47 // CRepositoryBackupStreamIndex::AddL
    48 //
    49 // Add a new repository UID and stream ID pair to the index
    50 void CRepositoryBackupStreamIndex::AddL(TUid aUid, TStreamId aSettingStreamId, TStreamId aDeletedSettingsStreamId, TStreamId aInstalledSettingsStreamId) 
    51 	{
    52 	TRepositoryBackupStreamIndexElement newIndexElement;
    53 	newIndexElement.Set(aUid,aSettingStreamId, aDeletedSettingsStreamId, aInstalledSettingsStreamId) ;
    54 	iStreamIndex.AppendL(newIndexElement);
    55 	}
    56 
    57 
    58 // 
    59 // CRepositoryBackupStreamIndex::GetNext
    60 //
    61 // Iterate through the index retrieving the next Reposirory UID and Stream ID
    62 // pair.
    63 //
    64 TInt CRepositoryBackupStreamIndex::GetNext(TUid& aUid, TStreamId& aSettingsStreamId, TStreamId& aDeletedSettingsStreamId, TStreamId& aInstalledSettingsStreamId)
    65 	{
    66 	TInt error = KErrNone ;
    67 	if (iIndex < iStreamIndex.Count())
    68 		{
    69 		iStreamIndex[iIndex++].Get(aUid, aSettingsStreamId, aDeletedSettingsStreamId, aInstalledSettingsStreamId) ;
    70 		}
    71 	else
    72 		{
    73 		error = KErrNotFound ;
    74 		}
    75 	return error ;
    76 	}
    77 
    78 
    79 //
    80 // Backup client class.
    81 //
    82 // Has Active object functionality to monitor the state of the publish and subscribe 
    83 // flags associated with backup and restore and also implements MActiveBackupDataClient
    84 // to perform active backup according to the "proxy data holder" model.
    85 //
    86 
    87 
    88 
    89 //
    90 // Usual 2 phase construction factory NewL NewLC classes
    91 //	
    92 CRepositoryBackupClient* CRepositoryBackupClient::NewLC(RFs& aFs) 
    93 	{
    94 	CRepositoryBackupClient* me = new(ELeave)CRepositoryBackupClient(aFs);
    95 	CleanupStack::PushL(me) ;
    96 	me->ConstructL() ;
    97 	return me ;
    98 	}
    99 	
   100 	
   101 CRepositoryBackupClient* CRepositoryBackupClient::NewL(RFs& aFs) 
   102 	{
   103 	CRepositoryBackupClient* me = CRepositoryBackupClient::NewLC(aFs) ;
   104 	CleanupStack::Pop(me) ;
   105 	return me ;
   106 	}
   107 	
   108 		
   109 		
   110 // 
   111 // Constructor - doesn't really do anything!
   112 //
   113 CRepositoryBackupClient::CRepositoryBackupClient(RFs& aFs) : CActive(EPriorityStandard), iFs(aFs), iRomScanDone(EFalse)
   114 	{
   115 	}
   116 
   117 
   118 // 
   119 // Phase 2 constructor
   120 //	
   121 void CRepositoryBackupClient::ConstructL()
   122 	{
   123 	// Create repository object
   124 	iRepository = new(ELeave) CServerRepository;
   125 	
   126 	// Notifier needed to open repositories.		
   127 	iNotifier = new(ELeave)CSessionNotifier ;
   128 
   129 	// Attach to Backup/Restore Pub/Sub property.	
   130 	User::LeaveIfError(iBackupRestoreProperty.Attach(KUidSystemCategory, KUidBackupRestoreKey)) ;
   131 	
   132 	// Add ourself to the active scheduler
   133 	CActiveScheduler::Add(this) ;
   134 	
   135 	// Initialise backup/restore status
   136 	iBackupStatus = ENoBackupActivty ;
   137 	
   138 	// Set active and request notification of changes to backup/restore
   139 	// Pub/Sub property.
   140 	StartL() ;
   141 	}
   142 
   143 
   144 
   145 //
   146 // Destructor
   147 //
   148 CRepositoryBackupClient::~CRepositoryBackupClient()
   149 	{
   150 	Cancel();
   151 	
   152 	iBackupRestoreProperty.Close() ;
   153 	
   154 	if (iRepository)
   155 		{
   156 		iRepository->Close();
   157 		delete(iRepository);
   158 		}
   159 		
   160 	if (iNotifier)
   161 		delete(iNotifier) ;
   162 	
   163 	if (iActiveBackupClient)
   164 		delete(iActiveBackupClient) ;	
   165 		
   166 	iRestoredRepositoriesArray.ResetAndDestroy();
   167 	iRestoredRepositoriesArray.Close();
   168 	
   169 	iFile.Close();
   170 	}
   171 
   172 
   173 
   174 
   175 // 
   176 // DoCancel - mandatory for active objects.
   177 //	
   178 void CRepositoryBackupClient::DoCancel()
   179 	{
   180 	iBackupRestoreProperty.Cancel() ;
   181 	}
   182 
   183 
   184 
   185 //
   186 // RunError
   187 //
   188 TInt CRepositoryBackupClient::RunError(TInt aError)
   189 	{
   190 	iRestoredRepositoriesArray.ResetAndDestroy();
   191 	UNUSED_VAR(aError);
   192 	return KErrNone;
   193 	}
   194 
   195 
   196 
   197 
   198 //
   199 // Test BUR Pub/Sub property set status, and notify BUR that we're
   200 // ready to go as appropriate. 
   201 //
   202 void CRepositoryBackupClient::TestBURstatusL(void)
   203 	{
   204 	TInt BURstatus ;
   205 	TRepositoryBackupState lastBackupStatus = ENoBackupActivty;
   206 	if (iBackupRestoreProperty.Get(BURstatus) != KErrNotFound)
   207 		{
   208 		BURstatus &= KBURPartTypeMask ;
   209 		switch (BURstatus)
   210 			{
   211 			case EBURUnset:	// State not yet set. Treat as no backup/restore in progress.
   212 			case EBURNormal:
   213 				// No backup or restore in progress. Probably
   214 				// means we've just completed an operation?
   215 				
   216 				lastBackupStatus = iBackupStatus;
   217 				iBackupStatus = ENoBackupActivty ;
   218 				
   219 				// Back to normal, so enable cache
   220 				TServerResources::iCacheManager->EnableCache();				
   221 				// delete the CActiveBackupClient
   222 				if (iActiveBackupClient)
   223 					{
   224 					delete iActiveBackupClient ;
   225 					iActiveBackupClient = NULL ;
   226 					}
   227 					
   228 				// Notify the changed keys if a restoration was just completed
   229 				if((lastBackupStatus == ERestoreInProgress)) 
   230 			    	{
   231 				    for(TInt i = 0; i < iRestoredRepositoriesArray.Count(); i++)
   232 				    	{
   233 						iRepository->OpenL(iRestoredRepositoriesArray[i]->Uid(), *iNotifier, EFalse);
   234 						iRepository->RestoreNotify(*iRestoredRepositoriesArray[i]);
   235 						iRepository->Close();
   236 						}
   237 					iRestoredRepositoriesArray.ResetAndDestroy();
   238 					}
   239 
   240 				break ;
   241 				
   242 			case EBURBackupFull :
   243 			case EBURBackupPartial :
   244 				// We don't distinguish between full and partial backups
   245 				// as the Backup engine will give us UIDs for all the
   246 				// repository data owners that want their stuff backed up
   247 				// anyway.
   248 				
   249 				// We don't want cache activity during backup
   250 				TServerResources::iCacheManager->DisableCache();
   251 				
   252 				// Any (and all!) repositories which have been opened in the
   253 				// course of system boot and normal operation will have been
   254 				// added to TServerResources::iOwnerIdLookUpTable as they were
   255 				// opened but there may well be repositories which need backing
   256 				// up and haven't yet been opened so we need to make sure that
   257 				// the lookup table is complete.
   258 				CompleteOwnerIdLookupTableL();
   259 
   260 				// Register with BUR engine
   261 				if (!iActiveBackupClient)
   262 					{
   263 					iActiveBackupClient = CActiveBackupClient::NewL(this) ;
   264 					}	
   265 				iActiveBackupClient->ConfirmReadyForBURL(KErrNone);
   266 				iBackupStatus = EBackupInProgress ;
   267 				break ;
   268 				
   269 			case EBURRestoreFull :
   270 			case EBURRestorePartial :
   271 				// We don't distinguish between full and partial restore
   272 				// either!
   273 				
   274 				// We don't want cache activity during restore either!
   275 				TServerResources::iCacheManager->DisableCache();
   276 				
   277 				// Register with BUR engine
   278 				if (!iActiveBackupClient)
   279 					{
   280 					iActiveBackupClient = CActiveBackupClient::NewL(this) ;
   281 					}
   282 				iActiveBackupClient->ConfirmReadyForBURL(KErrNone);
   283 				iBackupStatus = ERestoreInProgress ;
   284 				break ;
   285 							
   286 			}
   287 		}
   288 	}
   289 
   290 
   291 //
   292 // Request notification of changes in BUR Pub/Sub status
   293 //
   294 void CRepositoryBackupClient::StartL()
   295 	{
   296 	if (!IsActive())
   297 		{
   298 		TestBURstatusL();	
   299 		NotifyChange();
   300 		}
   301 	}
   302 
   303 
   304 //
   305 // Request notification of changes in BUR Pub/Sub status
   306 //
   307 void CRepositoryBackupClient::NotifyChange()
   308 	{
   309 	// Watch for changes in the property state. 
   310 	iBackupRestoreProperty.Subscribe(iStatus) ;
   311 	SetActive();
   312 	}
   313 
   314 	
   315 //
   316 // Handle changes of backup state through publish/subscribe
   317 //
   318 void CRepositoryBackupClient::RunL()
   319 	{	
   320 	NotifyChange() ;
   321 	TestBURstatusL();
   322 	}
   323 
   324 
   325 //
   326 // We can't estimate data size without A) having the SID of the data owner who's data
   327 // is to be backed up and B) going through the whole process of preparing the backup.
   328 //
   329 // The only sensible thing we can do is return an arbitrary value!
   330 //
   331 TUint CRepositoryBackupClient::GetExpectedDataSize(TDriveNumber /* aDrive */)
   332 	{
   333 	return KArbitraryNumber ;
   334 	}	
   335 
   336 
   337 
   338 
   339 //
   340 // Called by BUR engine to request a chunk of backup data.
   341 //	
   342 void CRepositoryBackupClient::GetBackupDataSectionL(TPtr8& aBuffer, TBool& aFinished)
   343 	{
   344 	const TInt chunkSize = aBuffer.MaxSize() ;
   345 
   346 	aFinished = EFalse ;
   347 		
   348 	// Pass a chunk of our prepared backup data in aBuffer
   349 	User::LeaveIfError(iFile.Read(aBuffer, chunkSize)) ;
   350 	TInt bytesRead = aBuffer.Length() ;
   351 
   352 	// Check to see if this was the last chunk of data.
   353 	if (bytesRead < chunkSize)
   354 		{
   355 		// Set "finished" flag so that BUR knows we're through...
   356 		aFinished = ETrue ;
   357 		
   358 		// ...and then tidy up by closing and deleting the backup file.
   359 		iFile.Close() ;
   360 		TParse backupFilePath ;
   361 		User::LeaveIfError(backupFilePath.Set(KBackupFileName, TServerResources::iBURDirectory, &KBackupFileExt));
   362 		TInt fileDeleteErr=iFs.Delete(backupFilePath.FullName()) ;
   363 		#ifdef _DEBUG
   364 			if (fileDeleteErr != KErrNone)
   365 			{
   366 				RDebug::Print(_L("CRepositoryBackupClient::GetBackupDataSectionL - Failed to delete file. Error = %d"), fileDeleteErr);
   367 			}
   368 		#else
   369 			UNUSED_VAR(fileDeleteErr);
   370 		#endif
   371 
   372 		}
   373 	}
   374 
   375 //
   376 // CRepositoryBackupClient::RestoreComplete
   377 //
   378 // Called when a Complete set of backup data has been received and written
   379 // to a file. We now need to open the file as a stream store, get the
   380 // index (list of repository UID and corresponding stream ID pairs, and then
   381 // reconstruct and save each repository in turn.
   382 // 
   383 void CRepositoryBackupClient::RestoreComplete(TDriveNumber /* aDrive */)
   384 	{
   385 	}
   386 
   387 
   388 void CRepositoryBackupClient::RestoreRepositoryAndListL(TUid repositoryUid, CDirectFileStore* store, TStreamId settingsStreamId, TStreamId deletedSettingsStreamId, TInt& repIndex)
   389     {
   390     // Add the restored repository to the restored repositories list.
   391     // Pass its changed-keys list to further restoring functions to add entries for post-restoration notification.
   392     repIndex = AddRestoredRepositoryL(repositoryUid);
   393     iRepository->RestoreRepositoryContentsL(*store, settingsStreamId, deletedSettingsStreamId, *iRestoredRepositoriesArray[repIndex]);
   394     iRepository->CommitChangesL();
   395     }
   396 
   397 	
   398 //
   399 // CRepositoryBackupClient::RestoreCompleteL
   400 //
   401 // Does the actual work of reconstructing repositories from backup data
   402 //
   403 //
   404 void CRepositoryBackupClient::RestoreCompleteL()
   405 	{
   406 	// All restore data recived so we can now recreate the repositories from the
   407 	// backup store
   408 	// Attempt to open the restore file as a CDirectFileStore
   409 	TParse restoreFilePath ;
   410 	User::LeaveIfError(restoreFilePath.Set(KRestoreFileName, TServerResources::iBURDirectory, &KRestoreFileExt));
   411 	CDirectFileStore* store = CDirectFileStore::OpenLC (iFs,restoreFilePath.FullName(), EFileRead|EFileShareReadersOnly);
   412 	if (store->Type()[0] != KDirectFileStoreLayoutUid)
   413 		{
   414 		// store wasn't quite what we were expecting - can't return an error, can't leave
   415 		// so all we can do is close the file, tidy up as best we can, and bail out!!!! 
   416 		CleanupStack::PopAndDestroy(store);
   417 		// If a debug build - record error
   418 		TInt fileDeleteErr=iFs.Delete(restoreFilePath.FullName()) ;
   419 		#ifdef _DEBUG
   420 			if (fileDeleteErr != KErrNone)
   421 			{
   422 				RDebug::Print(_L("CRepositoryBackupClient::RestoreCompleteL - Failed to delete file. Error = %d"), fileDeleteErr);
   423 			}
   424 		#else
   425 			UNUSED_VAR(fileDeleteErr);
   426 		#endif
   427 
   428 		User::Leave(KErrCorrupt);
   429 		}		
   430 
   431 
   432 	// Get the root stream and attempt to read a backup file header from it
   433 	TStreamId rootStreamId = store->Root() ;
   434 	RStoreReadStream rootStream ;
   435 	RStoreReadStream indexStream ;
   436 	rootStream.OpenLC(*store, rootStreamId);
   437 	TRepositoryBackupStreamHeader header ;
   438 	TRAPD(err, header.InternalizeL(rootStream)) ;
   439 	
   440 	// Check for a valid header by checking that the UID matches the UID
   441 	// of Central Repository and that the version number is sane.
   442 	if (err == KErrNotSupported)
   443 	    {
   444    		// Not a valid header - assume it's an old style backup stream,
   445 		// set extensions supported to none, set index stream to be
   446 		// root stream and reset read pointer to beginning.
   447 		iBackupExtensionsSupported = ENoBackupExtensions ;
   448 
   449 		CleanupStack::PopAndDestroy(&rootStream) ;
   450 		CleanupStack::PopAndDestroy(store) ;
   451 
   452 		// Try re-opening as old-style backup stream with index
   453 		// as root stream.		
   454 		CDirectFileStore* store = CDirectFileStore::OpenLC (iFs,restoreFilePath.FullName(), EFileRead|EFileShareReadersOnly);
   455 		indexStream.OpenLC(*store, rootStreamId) ;
   456 	    }
   457 	else
   458 		{
   459 	    // Got a valid header. Check for extensions supported by this
   460 	    // stream and get stream to read index from
   461 	    CleanupStack::PopAndDestroy(&rootStream) ;
   462 	    iBackupExtensionsSupported = header.getBackupExtensionsSupported();	
   463 	    TStreamId indexStreamId = header.getIndexStreamId() ;
   464 	    indexStream.OpenLC (*store, indexStreamId) ; 		
   465 		}
   466 	
   467 	CRepositoryBackupStreamIndex *restoreStreamIndex = CRepositoryBackupStreamIndex::NewLC();	
   468 	restoreStreamIndex->InternalizeL(indexStream, iBackupExtensionsSupported);
   469 
   470 	
   471 	// Iterate through index and attempt restore of each repository stream
   472 	// we find in it.
   473 	restoreStreamIndex->Reset();	
   474 	TUid repositoryUid;
   475 	TStreamId settingsStreamId(KNullStreamIdValue);
   476 	TStreamId deletedSettingsStreamId(KNullStreamIdValue);
   477 	TStreamId installedSettingsStreamId(KNullStreamIdValue);
   478 	
   479 	while (restoreStreamIndex->GetNext(repositoryUid, settingsStreamId, deletedSettingsStreamId, installedSettingsStreamId) == KErrNone)
   480 		{
   481 		iRepository->OpenL(repositoryUid, *iNotifier, EFalse);
   482 		iRepository->FailAllTransactions();
   483 		TInt repIndex;
   484 		TRAPD(err, RestoreRepositoryAndListL(repositoryUid, store, settingsStreamId, deletedSettingsStreamId, repIndex));
   485 		iRepository->Close();
   486 	    User::LeaveIfError(err);
   487 		// If the backup contains an installed repository containing default values for the settings, read them in
   488 		if (installedSettingsStreamId != KNullStreamId)
   489 			{
   490 			// create an empty repository in install directory, and restore the data from backup file
   491 			iRepository->RestoreInstallRepositoryL(repositoryUid, *store, installedSettingsStreamId, *iRestoredRepositoriesArray[repIndex]);
   492 			// remove the .ini install file (if exists) because it will clash with the restored file
   493 			TServerResources::DeleteCentrepFileL(repositoryUid, EInstall, EIni);
   494 			}
   495 		}
   496 
   497 	CleanupStack::PopAndDestroy(restoreStreamIndex) ;
   498 	CleanupStack::PopAndDestroy(&indexStream);
   499 	CleanupStack::PopAndDestroy(store);
   500 	// If a debug build - record error
   501 	TInt fileDeleteErr=iFs.Delete(restoreFilePath.FullName());
   502 	#ifdef _DEBUG
   503 		if (fileDeleteErr != KErrNone)
   504 		{
   505 			RDebug::Print(_L("CRepositoryBackupClient::RestoreCompleteL (2nd) - Failed to delete file. Error = %d"), fileDeleteErr);
   506 		}
   507 	#else
   508 		UNUSED_VAR(fileDeleteErr);
   509 	#endif
   510 
   511 	}
   512 	
   513 //
   514 // CRepositoryBackupClient::CompleteOwnerIdLookupTableL
   515 //
   516 // Open each repository in TServerResources::iDataDirectory.
   517 // Save the Rep UID and Owner Id of the rep to be used by
   518 // InitialiseGetProxyBackupDataL.
   519 void CRepositoryBackupClient::CompleteOwnerIdLookupTableL()
   520 	{
   521 
   522 	// Read contents of persist, install, and ROM directories and
   523 	// use them to build a list of repository candidates.
   524 	RArray <TUint32> repositoryList ;
   525 	CleanupClosePushL(repositoryList) ;
   526 
   527 	for (TBackupDirectoryScan scanDir = EScanRom; scanDir <= EScanPersist; scanDir = (TBackupDirectoryScan)(scanDir+1))
   528 		{
   529 		TPtrC directoryName ;
   530 		switch (scanDir)
   531 			{
   532 				case EScanRom :
   533 					if (TServerResources::iRomDirectory)
   534 						{
   535 						directoryName.Set(TServerResources::iRomDirectory->Des()) ;						
   536 						}
   537 					else
   538 						{
   539 						// if ROM directory doesn't exist or there are no files, skip scanning
   540 						continue;
   541 						}
   542 					break ;
   543 					
   544 				case EScanInstall :
   545 					directoryName.Set(TServerResources::iInstallDirectory->Des()) ;
   546 					break ;
   547 					
   548 				case EScanPersist :
   549 					directoryName.Set(TServerResources::iDataDirectory->Des()) ;
   550 					break ;				
   551 			}
   552 			
   553 		RDir dir;
   554 	    CleanupClosePushL(dir);
   555 		User::LeaveIfError(dir.Open(iFs, directoryName, KEntryAttNormal));
   556 	    
   557 	    TEntryArray dirEntries;
   558 	    TInt readError = KErrNone;
   559 	    
   560 		while (readError != KErrEof)  
   561 			{
   562 		    readError = dir.Read(dirEntries);
   563 	    
   564 		    if(readError != KErrNone && readError != KErrEof) 
   565 		    	{
   566 		    	User::Leave(readError);
   567 		    	}
   568 		    else
   569 		    	{
   570 		    	const TInt dirCount = dirEntries.Count();   
   571 		    	for (TInt i=0; i<dirCount; i++)
   572 		    		{
   573 		    		// Attempt to extract a repository UID from directory entry
   574 		    		TUid uid;
   575 					TInt insertionError;
   576 		    		if (KErrNone == TServerResources::GetUid(const_cast<TEntry&>(dirEntries[i]), uid))
   577 		    			{
   578 		    			insertionError=repositoryList.InsertInUnsignedKeyOrder(uid.iUid) ;
   579 						// Should leave in all cases other than KErrNone or KErrAlreadyExists						
   580 						if((insertionError != KErrNone) && (insertionError != KErrAlreadyExists ))
   581 							{
   582 							User::Leave(insertionError);
   583 							}						
   584 		    			}
   585 		    		}
   586 		    	}
   587 			}
   588 	    
   589 		CleanupStack::PopAndDestroy(&dir);
   590 		}
   591 		
   592 	// Open all repositories in turn. Save repository UID and owner ID
   593 	// in lookup table.
   594 	for(TInt i = 0; i<repositoryList.Count(); i++)
   595 		{
   596 		// Look to see if this repository already has an entry in the owner ID lookup table
   597 		if ( TServerResources::FindOwnerIdLookupMapping(repositoryList[i]) == KErrNotFound)
   598 			{
   599 			// Need to TRAP here as otherwise if ANY repository fails to open
   600 			// (e.g. due to corruption) it would cause the entire backup to
   601 			// fail
   602 			TRAPD(err, iRepository->OpenL(TUid::Uid(repositoryList[i]), *iNotifier));
   603 			if (err == KErrNoMemory)
   604 				{
   605 				User::Leave(err) ;
   606 				}
   607 			
   608 			else if (err == KErrNone)
   609 				{
   610 				// The act of opening a repository will cause it to add itself to the
   611 				// Repository/Owner UID mapping table so we don't need to do anything
   612 				// and can close it immediately!
   613 				iRepository->Close();
   614 				}
   615 			}
   616 		}
   617 		
   618 	CleanupStack::PopAndDestroy() ; // repositoryList	
   619 	}
   620 
   621 // CRepositoryBackupClient::InitialiseGetProxyBackupDataL
   622 //
   623 // Prepare data to be backed up. We get the Sid/Uid of the entity whos data
   624 // is to be backed up. What we do is to open each repository in turn (identified
   625 // by directory listing), check its owner, and if it matches the Sid/Uid we've
   626 // been given by secure backup externalise it to a stream within a file store.
   627 //
   628 void CRepositoryBackupClient::InitialiseGetProxyBackupDataL(TSecureId aSID, TDriveNumber /* aDrive */) 
   629 	{
   630 	// Prepare data for backup.
   631 
   632 	// Create file store
   633 	TParse backupFilePath ;
   634 	User::LeaveIfError(backupFilePath.Set(KBackupFileName, TServerResources::iBURDirectory, &KBackupFileExt));
   635 	CDirectFileStore* store = CDirectFileStore::ReplaceLC(iFs, backupFilePath.FullName(),
   636 	                                                      (EFileWrite | EFileShareExclusive));
   637 	const TUid uid2  = KNullUid ;	                                                     
   638 	store->SetTypeL(TUidType(KDirectFileStoreLayoutUid, uid2, KServerUid3)) ; 
   639 	
   640 	// Create a Backup Stream index
   641 	CRepositoryBackupStreamIndex* backupStreamIndex = CRepositoryBackupStreamIndex::NewLC();	
   642 	  
   643 	// Find the reps owned by aSID
   644 	for(TInt i = 0; i < TServerResources::iOwnerIdLookUpTable.Count(); ++i)
   645 		{
   646 		const TOwnerIdMapping& lookupTableEntry = TServerResources::iOwnerIdLookUpTable[i];
   647 		
   648 		if ( lookupTableEntry.iOwner == aSID )
   649 			{
   650 			TStreamId settingStreamId(KNullStreamIdValue);
   651 			TStreamId deletedSettingsStreamId(KNullStreamIdValue);
   652 			TStreamId installedSettingStreamId(KNullStreamIdValue);
   653 			// Found one match, open the repository and externalise content.
   654 			TUid uid = TUid::Uid(lookupTableEntry.iRepUid);
   655 			TRAPD(err,iRepository->OpenL(uid, *iNotifier));
   656 			if (err == KErrNoMemory)
   657 				{
   658 				User::Leave(err) ;
   659 				}
   660 			else if (err == KErrNone)
   661 				{			
   662 				iRepository->FailAllTransactions();
   663 				// externalise repository contents
   664 				iRepository->StoreRepositoryContentsL(*store, settingStreamId, deletedSettingsStreamId);
   665 				// Close repository.
   666 				iRepository->Close();
   667 				}
   668 				
   669 			TBool installExists=TServerResources::InstallFileExistsL(uid);
   670 			if(installExists)
   671 				{			
   672 				// load existing repository from install directory (default values installed post-build by SWI)
   673 				// and externalise installed repository contents				
   674 				TRAPD(err, iRepository->BackupInstallRepositoryL(uid, *store, installedSettingStreamId));
   675 				// We trap and discard most errors to be able to continue backing up other repositories in the list
   676 				if (err == KErrNoMemory)
   677 					{
   678 					User::Leave(err) ;
   679 					}
   680 				else if (err != KErrNone)
   681 					{
   682 					// If for any reason we haven't been able to back up the install repository, 
   683 					// we create an empty stream to preserve the format
   684 					installedSettingStreamId = KNullStreamIdValue;
   685 					}
   686 				}
   687 			// Add all to store index
   688 			backupStreamIndex->AddL(uid, settingStreamId, deletedSettingsStreamId, installedSettingStreamId) ;
   689 
   690 			}
   691 		}
   692 	// Write the stream index/dictionary as root stream within the store
   693 	// so we can access it when we do a restore later on
   694 	RStoreWriteStream indexStream ;
   695 	TStreamId indexStreamId = indexStream.CreateLC(*store) ;
   696 	backupStreamIndex->ExternalizeL(indexStream) ;
   697 	indexStream.CommitL() ;
   698 	CleanupStack::PopAndDestroy(&indexStream) ;
   699 	CleanupStack::PopAndDestroy (backupStreamIndex) ;
   700 		
   701 	// Create the Header and write it as the root stream within the store
   702 	// so we can access it when we do a restore later on
   703 	TRepositoryBackupStreamHeader header (indexStreamId) ;
   704 	RStoreWriteStream rootStream ;
   705 	TStreamId rootStreamId = rootStream.CreateLC(*store) ;
   706 	header.ExternalizeL(rootStream) ;
   707 	rootStream.CommitL() ;
   708 	
   709 	CleanupStack::PopAndDestroy(&rootStream) ;
   710 	store->SetRootL(rootStreamId);
   711 	store->CommitL();
   712 	CleanupStack::PopAndDestroy(store) ; 
   713 	
   714 	// Attempt to open file containing store ready to read back and send to
   715 	// BUR engine as a stream of bytes.
   716 	User::LeaveIfError(iFile.Open(iFs, backupFilePath.FullName(), (EFileRead | EFileShareExclusive))) ;
   717 	}
   718 
   719 
   720 
   721 
   722 //
   723 // CRepositoryBackupClient::InitialiseRestoreProxyBaseDataL
   724 //
   725 // Called when secure backup is about to start sending restore data.
   726 //
   727 void CRepositoryBackupClient::InitialiseRestoreProxyBaseDataL(TSecureId aSID, TDriveNumber /* aDrive*/)
   728 	{
   729 	// prepare for restore - Don't think we need to do anything here except prepare
   730 	// data structures to receive incoming data
   731 	
   732 	// Save SID so we can check that it corresponds with the owner information 
   733 	// in the restored data.
   734 	iSid = aSID ;
   735 	
   736 	// Open file to receive restored data
   737 	TParse restoreFilePath ;
   738 	User::LeaveIfError(restoreFilePath.Set(KRestoreFileName, TServerResources::iBURDirectory, &KRestoreFileExt));
   739 	User::LeaveIfError(iFile.Replace (iFs, restoreFilePath.FullName(), (EFileWrite | EFileShareExclusive)));
   740 	}
   741 
   742 
   743 
   744 
   745 //
   746 // CRepositoryBackupClient::RestoreBaseDataSectionL
   747 //
   748 // Called when secure backup has a chunk of restore data for us. Last data
   749 // segment identified by aFinished.
   750 //
   751 void CRepositoryBackupClient::RestoreBaseDataSectionL(TDesC8& aBuffer, TBool aFinished)
   752 	{
   753 	// Receive a chunk of restore data in aBuffer 
   754 	User::LeaveIfError(iFile.Write (aBuffer)) ;
   755 	if (aFinished)
   756 		{
   757 		iFile.Close() ;
   758 		
   759 		// All restore data recived so we can now recreate the repositories from the
   760 		// backup store	
   761 		RestoreCompleteL();	
   762 		}
   763 	}
   764 	
   765 void CRepositoryBackupClient::TerminateMultiStageOperation()
   766 	{
   767 	// Backup/Restore operation aborted!
   768 	// Tidy up all temporary data.
   769 	HBufC* burFileName = HBufC::New(KMaxFileName);
   770 	if(burFileName==NULL)
   771 		{
   772 		return;
   773 		}
   774 	TPtr burFileNamePtr(burFileName->Des());
   775 	iFile.FullName(burFileNamePtr); //get the full name of the temporary file
   776 	iFile.Close(); // close the file
   777 	// If a debug build - record error
   778 	TInt fileDeleteErr=iFs.Delete(burFileNamePtr);
   779 	#ifdef _DEBUG
   780 		if (fileDeleteErr != KErrNone)
   781 		{
   782 			RDebug::Print(_L("CRepositoryBackupClient::TerminateMultiStageOperation - Failed to delete file. Error = %d"), fileDeleteErr);
   783 		}
   784 	#else
   785 		UNUSED_VAR(fileDeleteErr);
   786 	#endif
   787 
   788 	delete burFileName;
   789 	}
   790 
   791 
   792 //
   793 // CRepositoryBackupClient::GetDataChecksum
   794 //
   795 // Not required and we don't implement it.
   796 //
   797 TUint CRepositoryBackupClient::GetDataChecksum(TDriveNumber /* aDrive */) 
   798 	{
   799 	return KArbitraryNumber;
   800 	}
   801 
   802 // 
   803 // CRepositoryBackupClient::GetSnapshotDataL
   804 //
   805 // Only required for incremental backup (which we don't support
   806 //
   807 void CRepositoryBackupClient::GetSnapshotDataL(TDriveNumber /* aDrive */, TPtr8& /* aBuffer */, TBool& /* aFinished */) 
   808 	{
   809 	User::Leave(KErrNotSupported) ;
   810 	}
   811 
   812 // 
   813 // CRepositoryBackupClient::InitialiseGetBackupDataL
   814 //
   815 // Used by "normal" active backup to prepare data - we use
   816 // InitialiseRestoreProxyBaseDataL so this shouldn't be called!
   817 //	
   818 void CRepositoryBackupClient::InitialiseGetBackupDataL(TDriveNumber /* aDrive */)
   819 	{
   820 	User::Leave(KErrNotSupported) ;
   821 	}
   822 	
   823 
   824 
   825 void CRepositoryBackupClient::InitialiseRestoreBaseDataL(TDriveNumber /* aDrive */)
   826 	{
   827 	// Check this! I Don't think this method should get called as we're a proxy
   828 	// and so implement InitialiseGetProxyBackupDataL!! 
   829 	User::Leave(KErrNotSupported) ;		
   830 	}
   831 	
   832 
   833 
   834 
   835 void CRepositoryBackupClient::InitialiseRestoreIncrementDataL(TDriveNumber /* aDrive */)
   836 	{
   837 	// Check this! I Don't think this method should get called as we're a proxy
   838 	// so don't do incremental backup!! 
   839 	User::Leave(KErrNotSupported) ;			
   840 	}
   841 
   842 
   843 
   844 void CRepositoryBackupClient::RestoreIncrementDataSectionL(TDesC8& /* aBuffer */, TBool /* aFinished */)
   845 	{
   846 	// Check this! I Don't think this method should get called as we're a proxy
   847 	// so don't do incremental backup!! 
   848 	User::Leave(KErrNotSupported) ;		
   849 	}
   850 
   851 
   852 //
   853 // Incremental backup isn't supported for the proxy data holder model so this
   854 // method should never be called.
   855 //
   856 // Not acceptable to leave even though it's an ...L function which we don't implement
   857 void CRepositoryBackupClient::AllSnapshotsSuppliedL() 
   858 	{
   859 	;
   860 	}
   861 
   862 
   863 
   864 //
   865 // Incremental backup not supported
   866 //	
   867 void CRepositoryBackupClient::ReceiveSnapshotDataL(TDriveNumber /* aDrive */, TDesC8& /* aBuffer */, TBool /* aLastSection */) 
   868 	{
   869 	User::Leave(KErrNotSupported) ;
   870 	}
   871 
   872 /**
   873 Adds a new entry to the restored repository list with the specified repository uid.
   874 
   875 Creates an object of the CRestoredRepository and inserts it to the list.
   876 If a repository of the specified uid is already on the list, the created object will 
   877 be deleted and no repeated entry will be inserted to the list.
   878 
   879 @param aUid The uid of the repository to be added to this array.
   880 @return The index of the new or existing entry within the list.
   881 */
   882 TInt CRepositoryBackupClient::AddRestoredRepositoryL(TUid aUid)
   883 	{
   884 	CRestoredRepository* rstRepos = new(ELeave) CRestoredRepository(aUid);
   885 	CleanupStack::PushL(rstRepos);
   886 	TInt err = iRestoredRepositoriesArray.InsertInOrder(rstRepos, TLinearOrder<CRestoredRepository>(CRestoredRepository::CompareUids));
   887 	if(err != KErrNone && err != KErrAlreadyExists)
   888 		User::Leave(err);
   889 	TInt index = iRestoredRepositoriesArray.FindInOrderL(rstRepos, TLinearOrder<CRestoredRepository>(CRestoredRepository::CompareUids));
   890 	CleanupStack::Pop();
   891 	
   892 	if(err == KErrAlreadyExists)
   893 		{
   894 		delete rstRepos;
   895 		}
   896 	return index;
   897 	}