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