os/persistentdata/persistentstorage/sql/SRC/Server/SqlBur.cpp
author sl@SLION-WIN7.fritz.box
Fri, 15 Jun 2012 03:10:57 +0200
changeset 0 bde4ae8d615e
permissions -rw-r--r--
First public contribution.
sl@0
     1
// Copyright (c) 2005-2010 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 "SqlBur.h"
sl@0
    17
#include "SqlAssert.h"
sl@0
    18
#include "OstTraceDefinitions.h"
sl@0
    19
#ifdef OST_TRACE_COMPILER_IN_USE
sl@0
    20
#include "SqlBurTraces.h"
sl@0
    21
#endif
sl@0
    22
#include "SqlTraceDef.h"
sl@0
    23
sl@0
    24
#define UNUSED_ARG(arg) arg = arg
sl@0
    25
sl@0
    26
_LIT(KSqlBurBackupExt,  ".bak");
sl@0
    27
_LIT(KSqlBurRestoreDir, "temprestore");
sl@0
    28
_LIT(KSqlBurAllFiles,   "*");
sl@0
    29
sl@0
    30
const TUint K8to16bitShift = 1;
sl@0
    31
sl@0
    32
//Extracts and returns 32-bit integer from aNumBuf buffer.
sl@0
    33
static TUint32 GetNumUint32L(const TDesC& aNumBuf)
sl@0
    34
	{
sl@0
    35
	TLex lex(aNumBuf);
sl@0
    36
	lex.SkipSpace();
sl@0
    37
	TUint32 num = 0xFFFFFFFF;
sl@0
    38
	__SQLLEAVE_IF_ERROR2(lex.Val(num, EHex));
sl@0
    39
	return num;
sl@0
    40
	}
sl@0
    41
sl@0
    42
//Extracts and returns 64-bit integer from aNumBuf buffer.
sl@0
    43
static TInt64 GetNumInt64L(const TDesC& aNumBuf)
sl@0
    44
	{
sl@0
    45
	TLex lex(aNumBuf);
sl@0
    46
	lex.SkipSpace();
sl@0
    47
	TInt64 num = -1;
sl@0
    48
	__SQLLEAVE_IF_ERROR2(lex.Val(num, EHex));
sl@0
    49
	return num;
sl@0
    50
	}
sl@0
    51
sl@0
    52
///////////////////////////////////////////////////////////////////////////////////////////////////////////////////
sl@0
    53
///////////////////////////////   CSqlBurEventMonitor    //////////////////////////////////////////////////////////
sl@0
    54
///////////////////////////////////////////////////////////////////////////////////////////////////////////////////
sl@0
    55
sl@0
    56
/** 
sl@0
    57
Standard two phase construction. Creates a CSqlBurEventMonitor instance.
sl@0
    58
@param aInterface A reference to an interface that is used for retrieving list of databases to be sent for backup.
sl@0
    59
@return An instance of the backup notifier
sl@0
    60
@leave  KErrNoMemory, an out of memory condition has occurred;
sl@0
    61
                      Note that the function may also leave with some other system-wide error codes.
sl@0
    62
*/
sl@0
    63
CSqlBurEventMonitor* CSqlBurEventMonitor::NewL(MSqlSrvBurInterface& aInterface)
sl@0
    64
	{
sl@0
    65
	CSqlBurEventMonitor* self = new (ELeave) CSqlBurEventMonitor(aInterface);
sl@0
    66
	CleanupStack::PushL(self);
sl@0
    67
	self->ConstructL();
sl@0
    68
	CleanupStack::Pop(self);
sl@0
    69
	SQL_TRACE_BUR(OstTrace1(TRACE_INTERNALS, CSQLBACKUPNOTIFIER_NEWL, "0x%X;CSqlBurEventMonitor::NewL", (TUint)self));
sl@0
    70
	return self;
sl@0
    71
	}
sl@0
    72
sl@0
    73
/**
sl@0
    74
Releases the allocated resources.
sl@0
    75
*/
sl@0
    76
CSqlBurEventMonitor::~CSqlBurEventMonitor()
sl@0
    77
	{
sl@0
    78
	SQL_TRACE_BUR(OstTrace1(TRACE_INTERNALS, CSQLBACKUPNOTIFIER_CSQLBACKUPNOTIFIER2, "0x%X;CSqlBurEventMonitor::~CSqlBurEventMonitor", (TUint)this));
sl@0
    79
	Cancel();
sl@0
    80
	iBurProperty.Close();
sl@0
    81
	DestroyContent();
sl@0
    82
	}
sl@0
    83
sl@0
    84
/**
sl@0
    85
Initializes data members with their default values. 
sl@0
    86
@param aInterface A reference to an interface that is used for retrieving list of databases to be sent for backup.
sl@0
    87
*/		
sl@0
    88
CSqlBurEventMonitor::CSqlBurEventMonitor(MSqlSrvBurInterface& aInterface) :
sl@0
    89
	CActive(EPriorityStandard), 
sl@0
    90
	iBurInterface(aInterface)
sl@0
    91
	{
sl@0
    92
	}
sl@0
    93
sl@0
    94
/**
sl@0
    95
Initializes the created CSqlBurEventMonitor object.  
sl@0
    96
@leave  KErrNoMemory, an out of memory condition has occurred;
sl@0
    97
                      Note that the function may also leave with some other system-wide error codes.
sl@0
    98
*/	
sl@0
    99
void CSqlBurEventMonitor::ConstructL()
sl@0
   100
	{
sl@0
   101
	__SQLLEAVE_IF_ERROR(iBurProperty.Attach(KSqlBurPropertyCategoryUid, KSqlBurBackupRestoreKey));
sl@0
   102
	CActiveScheduler::Add(this);
sl@0
   103
	iBurProperty.Subscribe(iStatus);
sl@0
   104
	SetActive();
sl@0
   105
	}
sl@0
   106
sl@0
   107
/** 
sl@0
   108
RunL() is called when the value of the {KUidSystemCategory, KUidBackupRestoreKey} gets changed.
sl@0
   109
That indicates: a backup or a restore is about to begin.
sl@0
   110
sl@0
   111
How the function works:
sl@0
   112
 - When a backup or restore notification is received, the function will subscribe again for notifications from
sl@0
   113
   the backup and restore property and will read the property status;
sl@0
   114
 - If the property status is conn::EBURUnset or conn::EBURNormal, the function will destroy iSqlBurCallback
sl@0
   115
   and iActiveBackupClient interfaces. No more callbacks will be reseived from the backup and restore server.
sl@0
   116
   This is the end of the backup or restore processing;
sl@0
   117
 - If the property status is conn::EBURBackupFull, conn::EBURBackupPartial, conn::EBURRestoreFull or 
sl@0
   118
   conn::EBURRestorePartial, the function will create iSqlBurCallback and iActiveBackupClient interface
sl@0
   119
   (iActiveBackupClient's NewL() receives iSqlBurCallback as an input parameter, registering this way the callback
sl@0
   120
   in the backup and restore server to be called later, when sending or retrieving data to/from the server).
sl@0
   121
   If the property read and the interface creation operations have been successful, the function will call 
sl@0
   122
   ConfirmReadyForBURL(KErrNone) to notify the backup and restore server that the SQL server is ready to send/retrieve 
sl@0
   123
   backup/restore data.
sl@0
   124
   If the current notification is that a backup is about to begin, after the confirmation the backup and restore server will
sl@0
   125
   call CSqlBurCallback::InitialiseGetProxyBackupDataL() once per {client secure id, drive} 
sl@0
   126
   followed by CSqlBurCallback::GetBackupDataSectionL() calls to retrieve the backup data. 
sl@0
   127
   If the current notification is that a restore is about to begin, after the confirmation the backup and restore server will
sl@0
   128
   call CSqlBurCallback::InitialiseRestoreProxyBaseDataL() once per {client secure id, drive} 
sl@0
   129
   followed by CSqlBurCallback::RestoreBaseDataSectionL() calls to send the restore data.
sl@0
   130
     
sl@0
   131
The current implementation has one design flaw. If a backup or restore notification is received, there are at lest 3
sl@0
   132
places before the ConfirmReadyForBURL() call, where the code may leave: 
sl@0
   133
 - the "property get" operation;
sl@0
   134
 - the iSqlBurCallback creation;
sl@0
   135
 - the iActiveBackupClient creation;
sl@0
   136
If a leave occurs at some of the mentioned places, that leave will be trapped by the current CActiveScheduler object
sl@0
   137
and CSqlBurEventMonitor::RunError() will be called with the leaved error code.
sl@0
   138
Problem #1: CSqlBurEventMonitor::RunError() won't do anything with the error (apart from printing a trace in the OST builds).
sl@0
   139
            The error is silently suppressed. The backup or restore won't start. But the client won't see any notification
sl@0
   140
            for that problem.
sl@0
   141
Problem #2: ConfirmReadyForBURL() won't be called. According to the backup and restore documentation, if
sl@0
   142
            ConfirmReadyForBURL() is called with KErrNone parameter, that's a confirmation for the backup and restore
sl@0
   143
            server to start the processing. If ConfirmReadyForBURL() is called with an error different than KErrNone,
sl@0
   144
            that's a confirmation for the backup and restore server that the client is not ready. No backup or restore
sl@0
   145
            will be started. The remote backup client will be notified about the problem.
sl@0
   146
After an investigation it was found that the same problems do exist in all active backup clients, none of them has 
sl@0
   147
solved the problems. Then, the code here will be kept as it is, it might be too dangerous to do a change right now.
sl@0
   148
sl@0
   149
@see CSqlBurEventMonitor::RunError()
sl@0
   150
@see CSqlBurCallback
sl@0
   151
@see CActiveBackupClient
sl@0
   152
@see CSqlBurCallback::InitialiseGetProxyBackupDataL()
sl@0
   153
@see CSqlBurCallback::GetBackupDataSectionL()
sl@0
   154
@see CSqlBurCallback::InitialiseRestoreProxyBaseDataL()
sl@0
   155
@see CSqlBurCallback::RestoreBaseDataSectionL()
sl@0
   156
sl@0
   157
@leave  KErrNoMemory, an out of memory condition has occurred;
sl@0
   158
                      Note that the function may also leave with some other system-wide error codes.
sl@0
   159
*/
sl@0
   160
void CSqlBurEventMonitor::RunL()
sl@0
   161
	{
sl@0
   162
	SQL_TRACE_BUR(OstTrace1(TRACE_INTERNALS, CSQLBACKUPNOTIFIER_RUNL_ENTRY, "Entry;0x%X;CSqlBurEventMonitor::RunL", (TUint)this));
sl@0
   163
	iBurProperty.Subscribe(iStatus);
sl@0
   164
	SetActive();
sl@0
   165
	TInt status;
sl@0
   166
	__SQLLEAVE_IF_ERROR(iBurProperty.Get(status));
sl@0
   167
	status &= conn::KBURPartTypeMask;
sl@0
   168
#ifdef _SQL_RDEBUG_PRINT
sl@0
   169
	SQL_TRACE_BUR(OstTraceExt2(TRACE_INTERNALS, CSQLBACKUPNOTIFIER_RUNL1, "0x%X;CSqlBurEventMonitor::RunL;status=%d", (TUint)this, status));
sl@0
   170
#else
sl@0
   171
	SQL_TRACE_BUR(OstTraceExt2(TRACE_INTERNALS, CSQLBACKUPNOTIFIER_RUNL2, "0x%X;CSqlBurEventMonitor::RunL;status=%{TBURPartType}", (TUint)this, status));
sl@0
   172
#endif	    
sl@0
   173
	switch(status)
sl@0
   174
		{
sl@0
   175
		case conn::EBURBackupFull:
sl@0
   176
		case conn::EBURBackupPartial:
sl@0
   177
		case conn::EBURRestoreFull:
sl@0
   178
		case conn::EBURRestorePartial:
sl@0
   179
			{
sl@0
   180
			// we only do full backups and full restores
sl@0
   181
			if(!(iSqlBurCallback && iActiveBackupClient))
sl@0
   182
				{
sl@0
   183
				DestroyContent();
sl@0
   184
				TRAPD(err, CreateContentL());
sl@0
   185
				if(err != KErrNone)
sl@0
   186
					{
sl@0
   187
					DestroyContent();
sl@0
   188
					__SQLLEAVE(err);
sl@0
   189
					}
sl@0
   190
				}
sl@0
   191
			iActiveBackupClient->ConfirmReadyForBURL(KErrNone);
sl@0
   192
			}
sl@0
   193
			break;
sl@0
   194
		//case conn::EBURUnset:
sl@0
   195
		//case conn::EBURNormal:
sl@0
   196
		default:
sl@0
   197
			DestroyContent();
sl@0
   198
			break;
sl@0
   199
		}
sl@0
   200
	SQL_TRACE_BUR(OstTrace1(TRACE_INTERNALS, CSQLBACKUPNOTIFIER_EXIT, "Exit;0x%X;CSqlBurEventMonitor::RunL", (TUint)this));
sl@0
   201
	SQL_BUR_TEST_STOP();
sl@0
   202
	}
sl@0
   203
sl@0
   204
/** 
sl@0
   205
Cancels the subscribtion for {KUidSystemCategory, KUidBackupRestoreKey} property changes.
sl@0
   206
*/
sl@0
   207
void CSqlBurEventMonitor::DoCancel()
sl@0
   208
	{
sl@0
   209
	iBurProperty.Cancel();
sl@0
   210
	}
sl@0
   211
sl@0
   212
/**
sl@0
   213
No-op. The method does nothing with the reported from CSqlBurEventMonitor::RunL() error  
sl@0
   214
(apart from logging a trace in OST builds).
sl@0
   215
Actually, the right action is to return KErrNone (as it is implemented), otherwise the default implementation of 
sl@0
   216
CActiveScheduler::Error() will panic the current thread.
sl@0
   217
sl@0
   218
@see CActiveScheduler::Error()
sl@0
   219
@see CSqlBurEventMonitor::RunL()
sl@0
   220
 
sl@0
   221
@return The RunL() error, if the RunL() call leaves.
sl@0
   222
@param The RunL() error
sl@0
   223
*/
sl@0
   224
TInt CSqlBurEventMonitor::RunError(TInt aError)
sl@0
   225
	{
sl@0
   226
	UNUSED_ARG(aError);
sl@0
   227
	SQL_TRACE_BUR(OstTraceExt2(TRACE_INTERNALS, CSQLBACKUPCLIENT_RUNERROR, "0x%X;CSqlBurEventMonitor::RunError;aError=%d", (TUint)this, aError));
sl@0
   228
	SQL_BUR_TEST_SET_ERROR(aError); 
sl@0
   229
	SQL_BUR_TEST_STOP();
sl@0
   230
	return KErrNone;
sl@0
   231
	}
sl@0
   232
sl@0
   233
/**
sl@0
   234
Creates iActiveBackupClient and iSqlBurCallback objects.
sl@0
   235
*/
sl@0
   236
void CSqlBurEventMonitor::CreateContentL()
sl@0
   237
	{
sl@0
   238
	iSqlBurCallback = CSqlBurCallback::NewL(iBurInterface);
sl@0
   239
	iActiveBackupClient = conn::CActiveBackupClient::NewL(iSqlBurCallback);
sl@0
   240
	}
sl@0
   241
sl@0
   242
/**
sl@0
   243
Destroys iActiveBackupClient and iSqlBurCallback objects.
sl@0
   244
*/
sl@0
   245
void CSqlBurEventMonitor::DestroyContent()
sl@0
   246
	{
sl@0
   247
	delete iActiveBackupClient;
sl@0
   248
	iActiveBackupClient = NULL;
sl@0
   249
	delete iSqlBurCallback;
sl@0
   250
	iSqlBurCallback = NULL;
sl@0
   251
	}
sl@0
   252
sl@0
   253
///////////////////////////////////////////////////////////////////////////////////////////////////////////////////
sl@0
   254
///////////////////////////////   CSqlBackupClient    /////////////////////////////////////////////////////////////
sl@0
   255
///////////////////////////////////////////////////////////////////////////////////////////////////////////////////
sl@0
   256
sl@0
   257
sl@0
   258
/** 
sl@0
   259
Creates new CSqlBurCallback instance.
sl@0
   260
The CSqlBurEventMonitor object monitors the state of the {KUidSystemCategory, KUidBackupRestoreKey} 
sl@0
   261
property. When a backup or a restore is about to begin, the CSqlBurEventMonitor object creates a
sl@0
   262
CSqlBurCallback instance, establishes a connection with the B&R server and passes a pointer to
sl@0
   263
the CSqlBurCallback callback to the BYR conenction.
sl@0
   264
The CSqlBurCallback methods will be called during the backup/restore for sending/retrieving data.
sl@0
   265
    
sl@0
   266
@param aInterface A reference to an interface that is used for retrieving list of databases to be sent for backup.
sl@0
   267
@return A pointer to the created CSqlBurCallback instance
sl@0
   268
@leave  KErrNoMemory, an out of memory condition has occurred;
sl@0
   269
                      Note that the function may also leave with some other system-wide error codes.
sl@0
   270
*/
sl@0
   271
CSqlBurCallback* CSqlBurCallback::NewL(MSqlSrvBurInterface& aInterface)
sl@0
   272
	{
sl@0
   273
	CSqlBurCallback* self = new (ELeave) CSqlBurCallback(aInterface);
sl@0
   274
	SQL_TRACE_BUR(OstTrace1(TRACE_INTERNALS, CSQLBACKUPCLIENT_NEWLC, "0x%X;CSqlBurCallback::NewL", (TUint)self));
sl@0
   275
	return self;
sl@0
   276
	}
sl@0
   277
sl@0
   278
/**
sl@0
   279
Initializes CSqlBurCallback data members with their default values. 
sl@0
   280
@param aInterface A reference to an interface that is used for retrieving list of databases to be sent for backup.
sl@0
   281
*/		
sl@0
   282
CSqlBurCallback::CSqlBurCallback(MSqlSrvBurInterface& aInterface) :
sl@0
   283
	iInterface(aInterface)
sl@0
   284
	{
sl@0
   285
	}
sl@0
   286
sl@0
   287
/** 
sl@0
   288
Releases the allocated resources.
sl@0
   289
*/
sl@0
   290
CSqlBurCallback::~CSqlBurCallback()
sl@0
   291
	{
sl@0
   292
	SQL_TRACE_BUR(OstTraceExt2(TRACE_INTERNALS, CSQLBACKUPCLIENT_CSQLBACKUPCLIENT2, "0x%X;CSqlBurCallback::~CSqlBurCallback;iFile.SubSessionHandle()=0x%X", (TUint)this, (TUint)iFile.SubSessionHandle()));
sl@0
   293
	BackupCleanup();
sl@0
   294
	(void)RestoreCleanup();
sl@0
   295
	}
sl@0
   296
sl@0
   297
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
sl@0
   298
/////////////////////////////////////       Full backup   //////////////////////////////////////////////////////////
sl@0
   299
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
sl@0
   300
sl@0
   301
/** 
sl@0
   302
This is called to let us know that the given SID is to be backed up.
sl@0
   303
We ask the SQL server (using iInterface, see MSqlSrvBurInterface for more details) 
sl@0
   304
for a list of databases that want to be backed up.
sl@0
   305
sl@0
   306
The backup is initiated by a notification received in CSqlBurEventMonitor::RunL() method.
sl@0
   307
InitialiseGetProxyBackupDataL() is called once per {client secure id, drive} and each 
sl@0
   308
InitialiseGetProxyBackupDataL() call is followed after that by a set of CSqlBurCallback::GetBackupDataSectionL() calls,
sl@0
   309
made from the backup and restore client dll.
sl@0
   310
sl@0
   311
During GetBackupDataSectionL() calls the CSqlBurCallback object will read the content of the databases from the list,
sl@0
   312
retrieved from the MSqlSrvBurInterface::GetBackUpListL() call and send the content to the backup and restore server.     
sl@0
   313
sl@0
   314
@see MSqlSrvBurInterface
sl@0
   315
@see CSqlBurEventMonitor::RunL()
sl@0
   316
@see CSqlBurCallback::GetBackupDataSectionL()
sl@0
   317
@see CSqlServer::GetBackUpListL() 
sl@0
   318
sl@0
   319
@param aSid the UID of the application to backup
sl@0
   320
@param aDrive the drive to be backed up
sl@0
   321
@leave  KErrNoMemory, an out of memory condition has occurred;
sl@0
   322
                      Note that the function may also leave with some other system-wide error codes.
sl@0
   323
*/
sl@0
   324
void CSqlBurCallback::InitialiseGetProxyBackupDataL(TSecureId aSid, TDriveNumber aDrive)
sl@0
   325
	{
sl@0
   326
	SQL_TRACE_BUR(OstTraceExt3(TRACE_INTERNALS, CSQLBACKUPCLIENT_INITIALIZEGETPROXYBACKUPDATAL, "0x%X;CSqlBurCallback::InitialiseGetProxyBackupDataL;aSid=0x%X;aDrive=%d", (TUint)this, (TUint)aSid.iId, (TInt)aDrive));
sl@0
   327
	BackupCleanup();
sl@0
   328
	iInterface.GetBackUpListL(aSid, aDrive, iFileList);
sl@0
   329
	iFileIndex = 0;
sl@0
   330
	iState = EBackupNoFileOpen;
sl@0
   331
	iBackupError = KErrNone;
sl@0
   332
	}
sl@0
   333
sl@0
   334
/** 
sl@0
   335
This is supposed to allow the B&R framework to know in advance how much
sl@0
   336
data is coming - but unfortunately there is no way to know this
sl@0
   337
at this stage since we don't even know yet what SID is being processed
sl@0
   338
So we just answer some number to make the BUE happy. It doesn't
sl@0
   339
actually rely on this number so there is no risk - the aFinishedFlag
sl@0
   340
indicates the end of data, not the value returned here. It is
sl@0
   341
supposed to allow the BUE to optimise its behaviour by know up front
sl@0
   342
the data volume.
sl@0
   343
sl@0
   344
@see CSqlBurEventMonitor::RunL()
sl@0
   345
@see CSqlBurCallback::InitialiseGetProxyBackupDataL()
sl@0
   346
sl@0
   347
@param aDrive Unused parameter (the drive number is logged in OST builds).
sl@0
   348
@return an arbitrary number (1024 at the moment)
sl@0
   349
*/
sl@0
   350
TUint CSqlBurCallback::GetExpectedDataSize(TDriveNumber aDrive)
sl@0
   351
	{
sl@0
   352
	UNUSED_ARG(aDrive);
sl@0
   353
	// we have no idea at this point - we even don't know who is to be backed up yet
sl@0
   354
	const TUint KArbitraryNumber = 1024;
sl@0
   355
	SQL_TRACE_BUR(OstTraceExt3(TRACE_INTERNALS, CSQLBACKUPCLIENT_GETEXPECTEDDATASIZE, "0x%X;CSqlBurCallback::GetExpectedDataSize;aDrive=%d;rc=%u", (TUint)this, (TInt)aDrive, KArbitraryNumber));
sl@0
   356
	return KArbitraryNumber;
sl@0
   357
	}
sl@0
   358
sl@0
   359
/** 
sl@0
   360
This is the backup state machine
sl@0
   361
Because the data has to be sent back in sections and the various
sl@0
   362
components of the dataflow may straddle chunks, we have to keep
sl@0
   363
track of where we are between each transfer - a state machine is
sl@0
   364
the simplest and most understandable implementation.
sl@0
   365
sl@0
   366
Please note how the function processes the errors occuring during the backup.
sl@0
   367
If an error occurs, the error is not propagated back to the B&R server immediatelly.
sl@0
   368
The error is stored in iBurError data member and is reported at the end of the backup process.
sl@0
   369
The reason for such unusual error reporting poicy is: the SQL server performs full backup of possibly more
sl@0
   370
than one database file. If an error occurs during the backup of the first file for example, the backup
sl@0
   371
process should not stop at that point. All files will be processed and then at the end, the error will be reproted.
sl@0
   372
sl@0
   373
In details, the function runs a state machine, where:
sl@0
   374
 - every file in the list retrieved in InitialiseGetProxyBackupDataL() is opened;
sl@0
   375
 - the file is read and 32-bit checksum over the file data - calculated;
sl@0
   376
 - a file backup header is prepared, including there the file size, file name, file name length, protocol verison number
sl@0
   377
   and the checksum. The header is sent to the backup restore server;
sl@0
   378
 - the file data is read and sent to the backup and restore server;
sl@0
   379
 - during the described above sequence no leave ever occurs. The error that occurs during the file processing,
sl@0
   380
   is stored into a data member of CSqlBurCallback class. At the end, after the last file in the list is processed,
sl@0
   381
   the backup and restore server will get a notification (via a User::Leave() call) regarding the error;
sl@0
   382
   The used error reporting policy allows all files to be process without interrupting the backup process.
sl@0
   383
   For example, if there are 5 files to be sent to the backup and restore server, an error that occurs during the 
sl@0
   384
   processing of file #3, won't prevent files #4 and #5 from being sent for backup.       
sl@0
   385
sl@0
   386
@see CSqlBurEventMonitor::RunL()
sl@0
   387
@see CSqlBurCallback::InitialiseGetProxyBackupDataL()
sl@0
   388
sl@0
   389
@param aBuffer Output parameter, the buffer where the data will be put to be passed back
sl@0
   390
@param aFinishedFlag Set to true when all data has been submitted for backup
sl@0
   391
@leave  KErrNoMemory, an out of memory condition has occurred;
sl@0
   392
                      Note that the function may also leave with some other system-wide error codes.
sl@0
   393
*/
sl@0
   394
void CSqlBurCallback::GetBackupDataSectionL(TPtr8& aBuffer, TBool& aFinishedFlag)
sl@0
   395
	{
sl@0
   396
	// don't assume they set it to false
sl@0
   397
	aFinishedFlag=EFalse;
sl@0
   398
	// any files to backup
sl@0
   399
	if(iFileList.Count()==0)
sl@0
   400
		{
sl@0
   401
		// nothing to backup
sl@0
   402
		SQL_TRACE_BUR(OstTrace1(TRACE_INTERNALS, CSQLBACKUPCLIENT_GETBACKUPDATASECTIONL1, "0x%X;CSqlBurCallback::GetBackupDataSectionL;file count is 0", (TUint)this));
sl@0
   403
		aFinishedFlag = ETrue;
sl@0
   404
		BackupCleanup();
sl@0
   405
		return;
sl@0
   406
		}
sl@0
   407
sl@0
   408
	// run the state machine
sl@0
   409
	for(TInt bufFreeSpace=aBuffer.MaxSize()-aBuffer.Size(); bufFreeSpace>0; bufFreeSpace=aBuffer.MaxSize()-aBuffer.Size())
sl@0
   410
		{
sl@0
   411
		switch(iState)
sl@0
   412
			{
sl@0
   413
			case EBackupNoFileOpen: // open a file for processing
sl@0
   414
				{
sl@0
   415
				if(iFileIndex>=iFileList.Count())
sl@0
   416
					{// all files have been processed - send the finished flag
sl@0
   417
					SQL_TRACE_BUR(OstTraceExt2(TRACE_INTERNALS, CSQLBACKUPCLIENT_GETBACKUPDATASECTIONL2, "0x%X;CSqlBurCallback::GetBackupDataSectionL;all files processed;iBackupError=%d", (TUint)this, iBackupError));
sl@0
   418
					aFinishedFlag = ETrue;
sl@0
   419
					BackupCleanup();
sl@0
   420
					__SQLLEAVE_IF_ERROR(iBackupError);
sl@0
   421
					return;
sl@0
   422
					}
sl@0
   423
				// open the database file to send
sl@0
   424
				TPtrC fname = iFileList[iFileIndex]->Des();
sl@0
   425
				TInt err = iFile.Open(iInterface.Fs(), fname, EFileRead | EFileShareExclusive);
sl@0
   426
				SQL_TRACE_BUR(OstTraceExt5(TRACE_INTERNALS, CSQLBACKUPCLIENT_GETBACKUPDATASECTIONL3, "0x%X;CSqlBurCallback::GetBackupDataSectionL;BEGIN;fname=%S;iFileIndex=%d;iFile.SubSessionHandle()=0x%X;err=%d", (TUint)this, __SQLPRNSTR(fname), iFileIndex, (TUint)iFile.SubSessionHandle(), err));
sl@0
   427
				if(KErrNone != err)
sl@0
   428
					{
sl@0
   429
					// there's nothing we can do if we can't open the file so we just skip it
sl@0
   430
					SetBackupError(err);
sl@0
   431
					++iFileIndex;
sl@0
   432
					break;
sl@0
   433
					}
sl@0
   434
				iState=EBackupOpenNothingSent;
sl@0
   435
				break;
sl@0
   436
				}
sl@0
   437
			case EBackupOpenNothingSent: // nothing sent (so far) for this file - send the header info
sl@0
   438
				{
sl@0
   439
				TInt64 fileSize;
sl@0
   440
				TInt err = iFile.Size(fileSize);
sl@0
   441
				if(KErrNone != err)
sl@0
   442
					{
sl@0
   443
					SetBackupError(err);
sl@0
   444
					iState = EBackupEndOfFile;
sl@0
   445
					break;
sl@0
   446
					}
sl@0
   447
				
sl@0
   448
				TUint64 checksum64 = 0;
sl@0
   449
				err = CheckSum(iFile, checksum64);
sl@0
   450
				SQL_TRACE_BUR(OstTraceExt3(TRACE_INTERNALS, CSQLBACKUPCLIENT_GETBACKUPDATASECTIONL4, "0x%X;CSqlBurCallback::GetBackupDataSectionL;CheckSum();iFileIndex=%d;err=%d", (TUint)this, iFileIndex, err));
sl@0
   451
				if(err != KErrNone)
sl@0
   452
					{
sl@0
   453
					//An error occured while reading the file (or there was not enough memory for the read buffer)
sl@0
   454
					SetBackupError(err);
sl@0
   455
					iState = EBackupEndOfFile;
sl@0
   456
					break;
sl@0
   457
					}
sl@0
   458
				// Only grab last 4 bytes of the checksum - enough to be satisfied that the backup and restore worked ok
sl@0
   459
				TUint32 checksum32 = checksum64 & KMaxTUint32;
sl@0
   460
sl@0
   461
                // build the header - this is an instance member because it
sl@0
   462
                // has to persist over multiple calls to this method
sl@0
   463
				TPtrC fname = iFileList[iFileIndex]->Des();
sl@0
   464
				iBuffer.Format(_L("%8x%8x%4x%16lx%8x%S"),
sl@0
   465
					checksum32,					// %8x
sl@0
   466
					KSqlBurMagicNum,			// %8x
sl@0
   467
					KSqlBurHeaderVersion,		// %4x
sl@0
   468
					fileSize,					// %16lx
sl@0
   469
					fname.Length(),				// %8x
sl@0
   470
					&fname);					// %S
sl@0
   471
				SQL_TRACE_BUR(OstTraceExt4(TRACE_INTERNALS, CSQLBACKUPCLIENT_GETBACKUPDATASECTIONL5, "0x%X;CSqlBackupClient::GetBackupDataSectionL;fileName=%S;hdrPtr=|%S|;fileSize=%lld", (TUint)this, __SQLPRNSTR(fname), __SQLPRNSTR(iBuffer), fileSize));
sl@0
   472
				
sl@0
   473
				// we need it to look like an 8bit buffer
sl@0
   474
				TPtr8 hdrPtr8((TUint8*)iBuffer.Ptr(), iBuffer.Size(), iBuffer.Size());
sl@0
   475
							
sl@0
   476
				TInt len = Min(hdrPtr8.Size(), bufFreeSpace);
sl@0
   477
				
sl@0
   478
				// append the header to the buffer (only till it's full)
sl@0
   479
				aBuffer.Append(hdrPtr8.Ptr(), len);
sl@0
   480
				
sl@0
   481
				// decide what needs to happen next
sl@0
   482
				// if complete then we need data, otherwise we need to put
sl@0
   483
				// the rest of the header in the next chunk
sl@0
   484
				if(hdrPtr8.Size() <= bufFreeSpace)
sl@0
   485
					{
sl@0
   486
					iState = EBackupOpenAllHeaderSent;
sl@0
   487
					}
sl@0
   488
				else
sl@0
   489
					{
sl@0
   490
					// we need to keep track of how much of the header has
sl@0
   491
					// been sent so that we only send the reminder on the next
sl@0
   492
					// iteration
sl@0
   493
					iHeaderSent = len;
sl@0
   494
					iState = EBackupOpenPartHeaderSent;
sl@0
   495
					}
sl@0
   496
				break;
sl@0
   497
				}
sl@0
   498
			case EBackupOpenPartHeaderSent: // need to send the rest of the header
sl@0
   499
				{
sl@0
   500
				// get back the header - this is already loaded with the necessary info
sl@0
   501
				// from the previous state we were in - EBackupOpenNothingSent
sl@0
   502
				
sl@0
   503
				// we need it to look like an 8bit buffer
sl@0
   504
				TPtr8 hdrPtr8((TUint8*)iBuffer.Ptr(), iBuffer.Size(), iBuffer.Size());
sl@0
   505
				
sl@0
   506
				// how many bytes have we yet to send?
sl@0
   507
				TInt bytesRemaining = hdrPtr8.Size() - iHeaderSent;
sl@0
   508
				TInt len = Min(bytesRemaining, bufFreeSpace);
sl@0
   509
				aBuffer.Append(hdrPtr8.Ptr() + iHeaderSent, len);
sl@0
   510
				
sl@0
   511
				if(bytesRemaining <= bufFreeSpace)
sl@0
   512
					{
sl@0
   513
					iHeaderSent = 0; // ready for next header
sl@0
   514
					iState = EBackupOpenAllHeaderSent;
sl@0
   515
					}
sl@0
   516
				else
sl@0
   517
					{
sl@0
   518
					iHeaderSent += len; // ready to do round again
sl@0
   519
					//iState=EBackupOpenPartHeaderSent; same state as now!
sl@0
   520
					}
sl@0
   521
				break;
sl@0
   522
				}
sl@0
   523
			case EBackupOpenAllHeaderSent: // need to send some data
sl@0
   524
				{
sl@0
   525
				TPtr8 ptr((TUint8*)aBuffer.Ptr() + aBuffer.Size(), 0, bufFreeSpace);
sl@0
   526
				TInt err = iFile.Read(ptr);
sl@0
   527
				if(err != KErrNone)
sl@0
   528
					{
sl@0
   529
					//An error occured while reading the file 
sl@0
   530
					SetBackupError(err);
sl@0
   531
					iState = EBackupEndOfFile;
sl@0
   532
					SQL_TRACE_BUR(OstTraceExt3(TRACE_INTERNALS, CSQLBACKUPCLIENT_GETBACKUPDATASECTIONL6, "0x%X;CSqlBurCallback::GetBackupDataSectionL;File read;iFileIndex=%d;err=%d", (TUint)this, iFileIndex, err));
sl@0
   533
					break;
sl@0
   534
					}
sl@0
   535
				TInt bytesRead = ptr.Size();
sl@0
   536
				aBuffer.SetLength(aBuffer.Size() + bytesRead);
sl@0
   537
				// EOF
sl@0
   538
				if(bytesRead == 0)
sl@0
   539
					{
sl@0
   540
					iState = EBackupEndOfFile;
sl@0
   541
					break;
sl@0
   542
					}
sl@0
   543
				break;
sl@0
   544
				}
sl@0
   545
			case EBackupEndOfFile:
sl@0
   546
				{
sl@0
   547
				SQL_TRACE_BUR(OstTraceExt3(TRACE_INTERNALS, CSQLBACKUPCLIENT_GETBACKUPDATASECTIONL7, "0x%X;CSqlBurCallback::GetBackupDataSectionL;END;iFile.SubSessionHandle()=0x%X;iFileIndex=%d", (TUint)this, (TUint)iFile.SubSessionHandle(), iFileIndex));
sl@0
   548
				iFile.Close();
sl@0
   549
				++iFileIndex; // move on to next file
sl@0
   550
				iState = EBackupNoFileOpen; // go round again
sl@0
   551
				break;
sl@0
   552
				}
sl@0
   553
			default:
sl@0
   554
				__ASSERT_DEBUG(EFalse, __SQLPANIC(ESqlPanicInternalError));
sl@0
   555
				break;
sl@0
   556
			}//end of the "switch" statement
sl@0
   557
		}//end of the "for" statement
sl@0
   558
	}
sl@0
   559
sl@0
   560
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
sl@0
   561
/////////////////////////////////////       Full restore   /////////////////////////////////////////////////////////
sl@0
   562
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
sl@0
   563
sl@0
   564
/** 
sl@0
   565
Called when the BUE wants to start sending data to us.
sl@0
   566
Creates the folder (if the folder does not exist) where the temporary files will be created during the restore process.
sl@0
   567
Deletes all files from the restore folder.
sl@0
   568
sl@0
   569
The restore is initiated by a notification received in CSqlBurEventMonitor::RunL() method.
sl@0
   570
InitialiseRestoreProxyBaseDataL() is called once per {client secure id, drive} and each 
sl@0
   571
InitialiseRestoreProxyBaseDataLL() call is followed after that by a set of CSqlBurCallback::RestoreBaseDataSectionL() 
sl@0
   572
calls, made from the backup and restore client dll.
sl@0
   573
sl@0
   574
During RestoreBaseDataSectionLL() calls the CSqlBurCallback object will receive data from the backup and resore server.     
sl@0
   575
sl@0
   576
@see CSqlBurEventMonitor::RunL()
sl@0
   577
@see CSqlBurCallback::RestoreBaseDataSectionL()
sl@0
   578
sl@0
   579
@param aSid the UID of the application that is to be restored. Not used (only logged in OST builds).
sl@0
   580
@param aDrive the drive to restore.
sl@0
   581
@leave  KErrNoMemory, an out of memory condition has occurred;
sl@0
   582
                      Note that the function may also leave with some other system-wide error codes.
sl@0
   583
*/
sl@0
   584
void CSqlBurCallback::InitialiseRestoreProxyBaseDataL(TSecureId aSid, TDriveNumber aDrive)
sl@0
   585
	{
sl@0
   586
	UNUSED_ARG(aSid);
sl@0
   587
	UNUSED_ARG(aDrive);
sl@0
   588
	SQL_TRACE_BUR(OstTraceExt3(TRACE_INTERNALS, CSQLBACKUPCLIENT_INITIALIZERESTOREPROXYBASEDATAL, "0x%X;CSqlBurCallback::InitialiseRestoreProxyBaseDataL;aSid=0x%X;aDrive=%d", (TUint)this, (TUint)aSid.iId, (TInt)aDrive));
sl@0
   589
	iBuffer.Zero();
sl@0
   590
	iState = ERestoreExpectChecksum;
sl@0
   591
	iRestoreDrive = aDrive;
sl@0
   592
	iRestoreId = aSid;
sl@0
   593
	//Create the directory for the temporary files created during the restore process.
sl@0
   594
	TFileName privatePath;
sl@0
   595
	__SQLLEAVE_IF_ERROR(iInterface.Fs().PrivatePath(privatePath));
sl@0
   596
	TDriveUnit driveUnit(iRestoreDrive);
sl@0
   597
	TDriveName driveName = driveUnit.Name();
sl@0
   598
	privatePath.Insert(0, driveName);
sl@0
   599
	__SQLLEAVE_IF_ERROR(iParse.Set(KSqlBurRestoreDir, &privatePath, 0));
sl@0
   600
	iRestoreDir.Copy(iParse.FullName());
sl@0
   601
	iRestoreDir.Append(KPathDelimiter);
sl@0
   602
	TInt err = iInterface.Fs().MkDirAll(iRestoreDir);
sl@0
   603
	if(err != KErrAlreadyExists)
sl@0
   604
		{
sl@0
   605
		__SQLLEAVE_IF_ERROR(err);
sl@0
   606
		}
sl@0
   607
	//Cleanup the restore directory
sl@0
   608
	err = RestoreCleanup();
sl@0
   609
	if(err != KErrNotFound)
sl@0
   610
		{
sl@0
   611
		__SQLLEAVE_IF_ERROR(err);
sl@0
   612
		}
sl@0
   613
	}
sl@0
   614
sl@0
   615
/** 
sl@0
   616
This is called by BUE when the restore has completed.
sl@0
   617
sl@0
   618
@see CSqlBurEventMonitor::RunL()
sl@0
   619
@see CSqlBurCallback::InitialiseRestoreProxyBaseDataL()
sl@0
   620
sl@0
   621
@param aDrive the drive that is being restored. Not used (only logged in OST builds).
sl@0
   622
*/
sl@0
   623
void CSqlBurCallback::RestoreComplete(TDriveNumber aDrive)
sl@0
   624
	{
sl@0
   625
	UNUSED_ARG(aDrive);
sl@0
   626
	SQL_TRACE_BUR(OstTraceExt2(TRACE_INTERNALS, CSQLBACKUPCLIENT_RESTORECOMPLETE, "0x%X;CSqlBurCallback::RestoreComplete;aDrive=%d", (TUint)this, (TInt)aDrive));
sl@0
   627
	iRestoreDrive = TDriveNumber(-1);
sl@0
   628
	iRestoreId = TSecureId(KNullUid);
sl@0
   629
	}
sl@0
   630
sl@0
   631
/** 
sl@0
   632
This is repeatedly called by the BUE to send us chunks of restore data (for the current SID)
sl@0
   633
Becuase the data is spread over chunks we need to manage the state across mutiple calls
sl@0
   634
to this method so we use a state machine.
sl@0
   635
sl@0
   636
The function runs the state machine and for each file block detected in the coming data, the function does:
sl@0
   637
 - creates a temporary file in the restore directory (created by InitialiseRestoreProxyBaseDataL());
sl@0
   638
 - stores the file data in the created temporary file;
sl@0
   639
 - During the 2 steps descirbed above, if an error occurs, that erro will be reproted to the backup and restore
sl@0
   640
   server (via a User::Leave() call);
sl@0
   641
 - When all data is received and stored in temporary files in the restore directory, 
sl@0
   642
   for each received file the function will:
sl@0
   643
   = move the original database file to the restore directory with a ".bak" extension added to the file name;
sl@0
   644
   = move the temporary file, which has the same name as the original database file, to the location of the
sl@0
   645
     original database file - the SQL server private data cage;
sl@0
   646
   = delete the file with the ".bak" extension;
sl@0
   647
   The three steps described above are implemented as "all or none" operation - if an error occurs during step (2),
sl@0
   648
   the content of the original database file will be restored from the file with the ".bak" extension.   
sl@0
   649
sl@0
   650
@see CSqlBurEventMonitor::RunL()
sl@0
   651
@see CSqlBurCallback::InitialiseRestoreProxyBaseDataL()
sl@0
   652
sl@0
   653
@param aInBuffer Buffer with data to be restored
sl@0
   654
@param aFinishedFlag Set when there is not more data to restore
sl@0
   655
@leave  KErrNoMemory, an out of memory condition has occurred;
sl@0
   656
                      Note that the function may also leave with some other system-wide error codes.
sl@0
   657
*/
sl@0
   658
void CSqlBurCallback::RestoreBaseDataSectionL(TDesC8& aInBuffer, TBool aFinishedFlag)
sl@0
   659
	{
sl@0
   660
	// used to walk the buffer
sl@0
   661
	// got a new buffer - because each time this method is called, we have a
sl@0
   662
	// fresh chunk of data
sl@0
   663
	TInt inBufferPos = 0;
sl@0
   664
sl@0
   665
	// to mark when the state machine is through
sl@0
   666
	TBool done = EFalse;
sl@0
   667
	
sl@0
   668
	// check whether this is an empty restore
sl@0
   669
	if(aFinishedFlag && aInBuffer.Size() == 0)
sl@0
   670
		{
sl@0
   671
		return;
sl@0
   672
		}
sl@0
   673
sl@0
   674
	TInt iterations = 0;
sl@0
   675
	
sl@0
   676
	// run the state machine
sl@0
   677
	do
sl@0
   678
		{
sl@0
   679
		// how many bytes are there available in the buffer for processing?
sl@0
   680
		TInt bytesAvailable = aInBuffer.Size() - inBufferPos;
sl@0
   681
		// the reason why we are testing finishedFlag is because we must
sl@0
   682
		// make sure we re-enter the machine to do the tidyup
sl@0
   683
		if(bytesAvailable <= 0 && !aFinishedFlag)
sl@0
   684
			{
sl@0
   685
			// ran out of data in the chunk
sl@0
   686
			// so we return and wait for more data to arrive
sl@0
   687
			return;
sl@0
   688
			}
sl@0
   689
		switch(iState)
sl@0
   690
			{
sl@0
   691
			case ERestoreExpectChecksum: // 16 bytes (the header is UTF16 encoded, 8 unicode characters for the checksum)
sl@0
   692
				{
sl@0
   693
				const TInt KCheckSumStrLen = 8;
sl@0
   694
				CopyBufData(aInBuffer, inBufferPos, iBuffer, KCheckSumStrLen);
sl@0
   695
				if(iBuffer.Length() == KCheckSumStrLen)
sl@0
   696
					{
sl@0
   697
					iChecksum = ::GetNumUint32L(iBuffer);
sl@0
   698
					iState = ERestoreExpectOldFileSize;
sl@0
   699
					iBuffer.Zero();
sl@0
   700
					}
sl@0
   701
				break;
sl@0
   702
				}
sl@0
   703
			case ERestoreExpectOldFileSize: // 16 bytes (the header is UTF16 encoded, 8 unicode characters for 32-bit old file size)
sl@0
   704
				{
sl@0
   705
				const TInt KOldFileSizeStrLen = 8;
sl@0
   706
				CopyBufData(aInBuffer, inBufferPos, iBuffer, KOldFileSizeStrLen);
sl@0
   707
				if(iBuffer.Length() == KOldFileSizeStrLen)
sl@0
   708
					{
sl@0
   709
					TUint32 oldFileSize = ::GetNumUint32L(iBuffer);
sl@0
   710
					if(oldFileSize == KSqlBurMagicNum)
sl@0
   711
						{
sl@0
   712
						iState = ERestoreExpectVersion;
sl@0
   713
						}
sl@0
   714
					else
sl@0
   715
						{
sl@0
   716
						iFileSize = oldFileSize;	
sl@0
   717
						iState = ERestoreExpectFileNameSize;
sl@0
   718
						}
sl@0
   719
					iBuffer.Zero();
sl@0
   720
					}
sl@0
   721
				break;
sl@0
   722
				}	
sl@0
   723
			case ERestoreExpectVersion:
sl@0
   724
				{
sl@0
   725
				const TInt KVersionStrLen = 4;
sl@0
   726
				CopyBufData(aInBuffer, inBufferPos, iBuffer, KVersionStrLen);
sl@0
   727
				if(iBuffer.Length() == KVersionStrLen)
sl@0
   728
					{
sl@0
   729
					//Ignore the version: ::GetNumUint32L(iBuffer);	
sl@0
   730
					//At this stage we know that the version is 2+
sl@0
   731
					iState = ERestoreExpectFileSize;
sl@0
   732
					iBuffer.Zero();
sl@0
   733
					}
sl@0
   734
				break;
sl@0
   735
				}
sl@0
   736
			case ERestoreExpectFileSize:
sl@0
   737
				{
sl@0
   738
				const TInt KFileSizeStrLen = 16;
sl@0
   739
				CopyBufData(aInBuffer, inBufferPos, iBuffer, KFileSizeStrLen);
sl@0
   740
				if(iBuffer.Length() == KFileSizeStrLen)
sl@0
   741
					{
sl@0
   742
					iFileSize = GetNumInt64L(iBuffer);	
sl@0
   743
					iState = ERestoreExpectFileNameSize;
sl@0
   744
					iBuffer.Zero();
sl@0
   745
					}
sl@0
   746
				SQL_TRACE_BUR(OstTraceExt2(TRACE_INTERNALS, CSQLBACKUPCLIENT_RESTOREBASEDATASECTONL1, "0x%X;CSqlBurCallback::RestoreBaseDataSectionL;iFileSize=%lld", (TUint)this, iFileSize));
sl@0
   747
				break;
sl@0
   748
				}
sl@0
   749
			case ERestoreExpectFileNameSize: // the size of the file name to restore
sl@0
   750
				{
sl@0
   751
				const TInt KFileNameLenStrLen = 8;
sl@0
   752
				CopyBufData(aInBuffer, inBufferPos, iBuffer, KFileNameLenStrLen);
sl@0
   753
				if(iBuffer.Length() == KFileNameLenStrLen)
sl@0
   754
					{
sl@0
   755
					iFileNameSize = GetNumUint32L(iBuffer);		
sl@0
   756
					iState = ERestoreExpectFileName;
sl@0
   757
					iBuffer.Zero();
sl@0
   758
					}
sl@0
   759
				break;
sl@0
   760
				}
sl@0
   761
			case ERestoreExpectFileName:  // the name of the file to restore
sl@0
   762
				{
sl@0
   763
				CopyBufData(aInBuffer, inBufferPos, iBuffer, iFileNameSize);
sl@0
   764
				SQL_TRACE_BUR(OstTraceExt4(TRACE_INTERNALS, CSQLBACKUPCLIENT_RESTOREBASEDATASECTONL2, "0x%X;CSqlBurCallback::RestoreBaseDataSectionL;BEGIN;iBuffer=%S;iBuffer.Length()=%d;iFileNameSize=%d", (TUint)this, __SQLPRNSTR(iBuffer), iBuffer.Length(), iFileNameSize));
sl@0
   765
				if(iBuffer.Length() == iFileNameSize)
sl@0
   766
					{
sl@0
   767
					iState = ERestoreExpectData;
sl@0
   768
					TParse parse;
sl@0
   769
					__SQLLEAVE_IF_ERROR(parse.Set(iBuffer, 0, 0));
sl@0
   770
					__SQLLEAVE_IF_ERROR(iParse.Set(parse.NameAndExt(), &iRestoreDir, 0));
sl@0
   771
					TPtrC fname(iParse.FullName());
sl@0
   772
					//The database is restored first to a temporary file, in the restore folder, on the same drive.
sl@0
   773
					__SQLLEAVE_IF_ERROR(iFile.Replace(iInterface.Fs(), fname, EFileWrite | EFileShareExclusive));
sl@0
   774
					SQL_TRACE_BUR(OstTraceExt3(TRACE_INTERNALS, CSQLBACKUPCLIENT_RESTOREBASEDATASECTONL3, "0x%X;CSqlBurCallback::RestoreBaseDataSectionL;fname=%S;iFile.SubSessionHandle()=0x%X", (TUint)this, __SQLPRNSTR(fname), (TUint)iFile.SubSessionHandle()));
sl@0
   775
					iBuffer.Zero();
sl@0
   776
					}
sl@0
   777
				break;
sl@0
   778
				}
sl@0
   779
			case ERestoreExpectData: // now for the data
sl@0
   780
				{
sl@0
   781
				TInt len = Min((aInBuffer.Size() - inBufferPos), iFileSize);
sl@0
   782
				TInt err = iFile.Write(aInBuffer.Mid(inBufferPos, len));
sl@0
   783
				if(err != KErrNone)
sl@0
   784
					{
sl@0
   785
					(void)RestoreCleanup();
sl@0
   786
					__SQLLEAVE(err);
sl@0
   787
					}
sl@0
   788
				inBufferPos += len;
sl@0
   789
				iFileSize -= len;
sl@0
   790
				if(iFileSize == 0)
sl@0
   791
					{
sl@0
   792
					iState = ERestoreComplete;
sl@0
   793
					}
sl@0
   794
				break;
sl@0
   795
				}
sl@0
   796
			case ERestoreComplete: // file completely restored
sl@0
   797
				{
sl@0
   798
				TUint64 checkSum64 = 0;
sl@0
   799
			    TInt restoreErr = iFile.Flush();
sl@0
   800
				if(restoreErr == KErrNone)
sl@0
   801
					{
sl@0
   802
					// calculate the checksum
sl@0
   803
					restoreErr = CheckSum(iFile, checkSum64);
sl@0
   804
					}
sl@0
   805
				iFile.Close();
sl@0
   806
				if(restoreErr != KErrNone)
sl@0
   807
					{
sl@0
   808
					(void)RestoreCleanup();
sl@0
   809
					__SQLLEAVE(restoreErr);
sl@0
   810
					}
sl@0
   811
				SQL_TRACE_BUR(OstTraceExt2(TRACE_INTERNALS, CSQLBACKUPCLIENT_RESTOREBASEDATASECTONL4, "0x%X;CSqlBurCallback::RestoreBaseDataSectionL;END;iFile.SubSessionHandle()=0x%X", (TUint)this, (TUint)iFile.SubSessionHandle()));
sl@0
   812
				TUint32 checkSum32 = checkSum64 & KMaxTUint32;
sl@0
   813
                if(checkSum32 != iChecksum)
sl@0
   814
                    {
sl@0
   815
					(void)RestoreCleanup();
sl@0
   816
                    __SQLLEAVE(KErrCorrupt);
sl@0
   817
                    }
sl@0
   818
				if((aInBuffer.Size() - inBufferPos) > 0)
sl@0
   819
					{//There are bytes to be consumed in the input buffer
sl@0
   820
					iState = ERestoreExpectChecksum;
sl@0
   821
					break;
sl@0
   822
					}
sl@0
   823
				SQL_TRACE_BUR(OstTrace1(TRACE_INTERNALS, CSQLBACKUPCLIENT_RESTOREBASEDATASECTONL5, "0x%X;CSqlBurCallback::RestoreBaseDataSectionL;aFinishedFlag=ETrue", (TUint)this));
sl@0
   824
				//End of data. We have all data restored in the restore folder.
sl@0
   825
				//The final step of the "restoring files" process consists of the following sub-steps:
sl@0
   826
				// - Rename the database file to be restored to a file with ".bak" extension
sl@0
   827
				// - Rename the file with the restored data to the database file
sl@0
   828
				// - Delete the file with ".bak" extension
sl@0
   829
				//Do not leave during the restore process! Restore as much files as possible.
sl@0
   830
				//The only excpetion is TParse::Set() - if it fails it is a fatal error, the
sl@0
   831
				//restored file path cannot be constructed.
sl@0
   832
				__ASSERT_DEBUG(iRestoreDrive != TDriveNumber(-1), __SQLPANIC(ESqlPanicInternalError));
sl@0
   833
				__ASSERT_DEBUG(iRestoreId != TSecureId(KNullUid), __SQLPANIC(ESqlPanicInternalError));
sl@0
   834
				//Include the aUid and the "*" mask
sl@0
   835
				TUidName uidName = (static_cast <TUid> (iRestoreId)).Name();
sl@0
   836
				TBuf<KMaxUidName + sizeof(KSqlBurAllFiles)> fileNameMask(uidName);
sl@0
   837
				fileNameMask.Append(KSqlBurAllFiles);
sl@0
   838
				__SQLLEAVE_IF_ERROR(iParse.Set(fileNameMask, &iRestoreDir, 0));
sl@0
   839
				CDir* dir = NULL;
sl@0
   840
				TPtrC searchPattern(iParse.FullName());
sl@0
   841
				SQL_TRACE_BUR(OstTraceExt2(TRACE_INTERNALS, CSQLBACKUPCLIENT_RESTOREBASEDATASECTONL55, "0x%X;CSqlBurCallback::RestoreBaseDataSectionL;search pattern=%S", (TUint)this, __SQLPRNSTR(searchPattern)));
sl@0
   842
				restoreErr = iInterface.Fs().GetDir(searchPattern, KEntryAttNormal, ESortNone, dir);
sl@0
   843
				if(restoreErr == KErrNone)
sl@0
   844
					{
sl@0
   845
					SQL_TRACE_BUR(OstTraceExt2(TRACE_INTERNALS, CSQLBACKUPCLIENT_RESTOREBASEDATASECTONL6, "0x%X;CSqlBurCallback::RestoreBaseDataSectionL;restored files=%d", (TUint)this, dir->Count()));
sl@0
   846
					for(TInt i=0;i<dir->Count();++i)
sl@0
   847
						{
sl@0
   848
						const TEntry& entry = (*dir)[i];
sl@0
   849
						__SQLLEAVE_IF_ERROR(iParse.Set(entry.iName, &iRestoreDir, 0));
sl@0
   850
						TFileName dbName(iParse.FullName());
sl@0
   851
						SQL_TRACE_BUR(OstTraceExt2(TRACE_INTERNALS, CSQLBACKUPCLIENT_RESTOREBASEDATASECTONL7, "0x%X;CSqlBurCallback::RestoreBaseDataSectionL;restored file=%S", (TUint)this, __SQLPRNSTR(dbName)));
sl@0
   852
						TInt pos = dbName.Find(KSqlBurRestoreDir);
sl@0
   853
						__ASSERT_DEBUG(pos >= 0, __SQLPANIC(ESqlPanicInternalError));
sl@0
   854
						dbName.Delete(pos, KSqlBurRestoreDir().Length() + 1);//"+1" for the path delimitier
sl@0
   855
						SQL_TRACE_BUR(OstTraceExt2(TRACE_INTERNALS, CSQLBACKUPCLIENT_RESTOREBASEDATASECTONL8, "0x%X;CSqlBurCallback::RestoreBaseDataSectionL;database=%S", (TUint)this, __SQLPRNSTR(dbName)));
sl@0
   856
						TFileName bakDbName(iParse.FullName());
sl@0
   857
						bakDbName.Append(KSqlBurBackupExt);
sl@0
   858
						SQL_TRACE_BUR(OstTraceExt2(TRACE_INTERNALS, CSQLBACKUPCLIENT_RESTOREBASEDATASECTONL9, "0x%X;CSqlBurCallback::RestoreBaseDataSectionL;backup file=%S", (TUint)this, __SQLPRNSTR(dbName)));
sl@0
   859
						//Now, dbName contains the original database (full path), iParse - the restored file,
sl@0
   860
						//bakDbName - backup file name
sl@0
   861
						TInt err = iInterface.Fs().Rename(dbName, bakDbName);
sl@0
   862
						if(err == KErrNone || err == KErrNotFound)
sl@0
   863
							{
sl@0
   864
							err = iInterface.Fs().Rename(iParse.FullName(), dbName);
sl@0
   865
							if(err == KErrNone)
sl@0
   866
								{//commit: delete the backup database file
sl@0
   867
								SQL_TRACE_BUR(OstTraceExt2(TRACE_INTERNALS, CSQLBACKUPCLIENT_RESTOREBASEDATASECTONL10, "0x%X;CSqlBurCallback::RestoreBaseDataSectionL;Commit;file=%S", (TUint)this, __SQLPRNSTR(dbName)));
sl@0
   868
								(void)iInterface.Fs().Delete(bakDbName);
sl@0
   869
								}
sl@0
   870
							else
sl@0
   871
								{//rollback: restore the original database file
sl@0
   872
								err = iInterface.Fs().Rename(bakDbName, dbName);
sl@0
   873
								SQL_TRACE_BUR(OstTraceExt3(TRACE_INTERNALS, CSQLBACKUPCLIENT_RESTOREBASEDATASECTONL11, "0x%X;CSqlBurCallback::RestoreBaseDataSectionL;Rollback;file=%S;err=%d", (TUint)this, __SQLPRNSTR(dbName), err));
sl@0
   874
								}
sl@0
   875
							}
sl@0
   876
						if(err != KErrNone && err != KErrNotFound)
sl@0
   877
							{
sl@0
   878
							if(restoreErr == KErrNone)
sl@0
   879
								{
sl@0
   880
								restoreErr = err;
sl@0
   881
								}
sl@0
   882
							}
sl@0
   883
						}//for(...)
sl@0
   884
					delete dir;
sl@0
   885
					}//iInterface.Fs().GetDir(...)
sl@0
   886
				done = ETrue;
sl@0
   887
				(void)RestoreCleanup();
sl@0
   888
				if(restoreErr != KErrNone)
sl@0
   889
					{
sl@0
   890
					__SQLLEAVE(restoreErr);
sl@0
   891
					}
sl@0
   892
				break;
sl@0
   893
				}
sl@0
   894
			default:
sl@0
   895
				__ASSERT_DEBUG(EFalse, __SQLPANIC(ESqlPanicInternalError));
sl@0
   896
				break;
sl@0
   897
			}//switch(...)
sl@0
   898
		if((aInBuffer.Size() - inBufferPos) == bytesAvailable)
sl@0
   899
			{//No bytes have been consumed from the buffer. 
sl@0
   900
			if(++iterations > 1 && !done)
sl@0
   901
				{//This is the second iteration in the loop where no bytes have been consumed from the input buffer. 
sl@0
   902
				 //But the "done" flag is still false. Corrupted archive.
sl@0
   903
				__SQLLEAVE(KErrCorrupt);
sl@0
   904
				}
sl@0
   905
			}
sl@0
   906
		} while(!done);
sl@0
   907
	}
sl@0
   908
sl@0
   909
/** 
sl@0
   910
The operation was terminated - we should tidyup here (as best we can)
sl@0
   911
Backup: close the file, free the allocated memory for the file names.
sl@0
   912
Restore: since the final restore step is a non-leaving one, nothing special needs to be done here - 
sl@0
   913
RestoreCleanup() is called to close the file and delete if there are any temporary files left.
sl@0
   914
*/	
sl@0
   915
void CSqlBurCallback::TerminateMultiStageOperation()
sl@0
   916
	{
sl@0
   917
	BackupCleanup();
sl@0
   918
	(void)RestoreCleanup();
sl@0
   919
	}
sl@0
   920
sl@0
   921
/** 
sl@0
   922
We do our own checksumming so we don't need this
sl@0
   923
@return the checksum
sl@0
   924
@param aDrive the drive affected (unused)
sl@0
   925
*/
sl@0
   926
TUint CSqlBurCallback::GetDataChecksum(TDriveNumber /* aDrive */)
sl@0
   927
	{
sl@0
   928
	// not required - not implemented
sl@0
   929
	const TUint KArbitraryNumber = 1024;
sl@0
   930
	return KArbitraryNumber;
sl@0
   931
	}
sl@0
   932
sl@0
   933
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
sl@0
   934
///////////////////////////     Incremental backup/restore      ////////////////////////////////////////////////////
sl@0
   935
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
sl@0
   936
sl@0
   937
/** 
sl@0
   938
We don't support incremental backup
sl@0
   939
*/
sl@0
   940
void CSqlBurCallback::GetSnapshotDataL(TDriveNumber /* aDrive */, TPtr8& /* aBuffer */, TBool& /* aFinishedFlag */)
sl@0
   941
	{
sl@0
   942
	__SQLLEAVE(KErrNotSupported);
sl@0
   943
	}
sl@0
   944
sl@0
   945
/** 
sl@0
   946
We don't support incremental backup
sl@0
   947
*/
sl@0
   948
void CSqlBurCallback::InitialiseGetBackupDataL(TDriveNumber /* aDrive */)
sl@0
   949
	{
sl@0
   950
	__SQLLEAVE(KErrNotSupported);
sl@0
   951
	}
sl@0
   952
sl@0
   953
/** 
sl@0
   954
We don't support incremental backup
sl@0
   955
*/
sl@0
   956
void CSqlBurCallback::InitialiseRestoreBaseDataL(TDriveNumber /* aDrive */)
sl@0
   957
	{
sl@0
   958
	__SQLLEAVE(KErrNotSupported);
sl@0
   959
	}
sl@0
   960
sl@0
   961
/** 
sl@0
   962
We don't support incremental backup
sl@0
   963
*/
sl@0
   964
void CSqlBurCallback::InitialiseRestoreIncrementDataL(TDriveNumber /* aDrive */)
sl@0
   965
	{
sl@0
   966
	__SQLLEAVE(KErrNotSupported);
sl@0
   967
	}
sl@0
   968
sl@0
   969
/** 
sl@0
   970
We don't support incremental backup
sl@0
   971
*/
sl@0
   972
void CSqlBurCallback::RestoreIncrementDataSectionL(TDesC8& /* aBuffer */, TBool /* aFinishedFlag */)
sl@0
   973
	{
sl@0
   974
	__SQLLEAVE(KErrNotSupported);
sl@0
   975
	}
sl@0
   976
sl@0
   977
/** 
sl@0
   978
We don't support incremental backup
sl@0
   979
*/
sl@0
   980
void CSqlBurCallback::AllSnapshotsSuppliedL()
sl@0
   981
	{
sl@0
   982
	}
sl@0
   983
sl@0
   984
/** 
sl@0
   985
We don't support incremental backup
sl@0
   986
*/
sl@0
   987
void CSqlBurCallback::ReceiveSnapshotDataL(TDriveNumber /* aDrive */, TDesC8& /* aBuffer */, TBool /* aFinishedFlag */)
sl@0
   988
	{
sl@0
   989
	__SQLLEAVE(KErrNotSupported);
sl@0
   990
	}
sl@0
   991
sl@0
   992
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
sl@0
   993
///////////////////////////     Helper functions      //////////////////////////////////////////////////////////////
sl@0
   994
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
sl@0
   995
sl@0
   996
/** 
sl@0
   997
A simple checksumming algorithm to allow a degree
sl@0
   998
of trust that the backup and restore worked.
sl@0
   999
Note the file pointer will be back at the start when the function call completes successfully.
sl@0
  1000
In case of an error, the position of the file pointer is undetermined.
sl@0
  1001
sl@0
  1002
@param aOpenFile Already opened database file on which the checksum is calculated.
sl@0
  1003
@param aCheckSum Output parameter. The checksum is returned in this parameter.
sl@0
  1004
@return KErrNoMemory, an out of memory condition has occurred;
sl@0
  1005
                      Note that the function may also return some other system-wide error codes.
sl@0
  1006
*/
sl@0
  1007
TInt CSqlBurCallback::CheckSum(const RFile64& aOpenFile, TUint64& aCheckSum) const
sl@0
  1008
	{
sl@0
  1009
	// scoot through the database file building the checksum
sl@0
  1010
	aCheckSum = 0;
sl@0
  1011
	TInt64 seekPos = 0; // rewind first
sl@0
  1012
	TInt err = aOpenFile.Seek(ESeekStart, seekPos);
sl@0
  1013
	if(err != KErrNone)
sl@0
  1014
		{
sl@0
  1015
		return err;
sl@0
  1016
		}
sl@0
  1017
	const TUint KCheckSumBlockSize = 4 * 1024;
sl@0
  1018
	HBufC8* buf = HBufC8::New(KCheckSumBlockSize);
sl@0
  1019
	if(!buf)
sl@0
  1020
		{
sl@0
  1021
		return KErrNoMemory;
sl@0
  1022
		}
sl@0
  1023
	TPtr8 ptr = buf->Des();
sl@0
  1024
	for(;;)
sl@0
  1025
		{
sl@0
  1026
		err = aOpenFile.Read(ptr);
sl@0
  1027
		if(err != KErrNone)
sl@0
  1028
			{
sl@0
  1029
			delete buf;
sl@0
  1030
			return err;
sl@0
  1031
			}
sl@0
  1032
		TInt len = ptr.Length();
sl@0
  1033
		if(len == 0)
sl@0
  1034
			{
sl@0
  1035
			break;
sl@0
  1036
			}
sl@0
  1037
		// calculate the checksum
sl@0
  1038
		for(TInt i=0;i<len;++i)
sl@0
  1039
			{
sl@0
  1040
			aCheckSum = (aCheckSum << 1) | (aCheckSum >> 63);
sl@0
  1041
			aCheckSum += ptr[i];
sl@0
  1042
 			}
sl@0
  1043
		};		
sl@0
  1044
	delete buf;
sl@0
  1045
	// restore file position
sl@0
  1046
	seekPos = 0;
sl@0
  1047
	err = aOpenFile.Seek(ESeekStart,seekPos);
sl@0
  1048
	return err;
sl@0
  1049
	}
sl@0
  1050
sl@0
  1051
/**
sl@0
  1052
Reads the content of aInBuf from position aInBufReadPos and stores the data into aOutBuf.
sl@0
  1053
aDataLen is the length of the data. If the input buffer does not contain all the data, then only the
sl@0
  1054
available data will be copied to the output buffer.
sl@0
  1055
sl@0
  1056
How the function works. It is called during the restore process and aInBuf parameter contains a block of raw
sl@0
  1057
data sent by the B&R server. The calling function, RestoreBaseDataSectionL(), uses a state 
sl@0
  1058
machine to processes the incoming data. At particular moment RestoreBaseDataSectionL() will process the data header 
sl@0
  1059
and will have to read "aDataLen" 16-bit characters at position "aInBufReadPos". If there are "aDataLen" characters
sl@0
  1060
at position "aInBufReadPos" and enough free space in "aOutBuf", CopyBufData() will copy all of them,  
sl@0
  1061
otherwise CopyBufData() will copy as much characters as possible (in which case RestoreBaseDataSectionL() will
sl@0
  1062
stay in the same state, waiting for more data from the B&R server).
sl@0
  1063
sl@0
  1064
@param aInBuf        8-bit buffer with input data
sl@0
  1065
@param aInBufReadPos The position in the buffer from which the read operation starts. 
sl@0
  1066
                     When the "buffer read" operatio completes, aInBufReadPos is updated with the 
sl@0
  1067
                     number of bytes read from the input buffer. 
sl@0
  1068
@param aOutBuf       16-bit output buffer. The data read from the input buffer is stored in the output buffer.
sl@0
  1069
@param aDataLen      How much bytes to be read from the input buffer. Note that if there is not enough
sl@0
  1070
                     data in the input buffer, the function will read as much as possible from the input buffer.
sl@0
  1071
                     The aInBufReadPos in/out parameter will be updated with the actual number of bytes read.                     
sl@0
  1072
*/
sl@0
  1073
void CSqlBurCallback::CopyBufData(const TDesC8& aInBuf, TInt& aInBufReadPos, TDes& aOutBuf, TInt aDataLen)
sl@0
  1074
	{
sl@0
  1075
    __ASSERT_DEBUG(aInBufReadPos >= 0, __SQLPANIC(ESqlPanicBadArgument));
sl@0
  1076
    __ASSERT_DEBUG(aDataLen > 0, __SQLPANIC(ESqlPanicBadArgument));
sl@0
  1077
	
sl@0
  1078
	TInt needed = (aDataLen - aOutBuf.Length()) << K8to16bitShift;
sl@0
  1079
	TInt available = aInBuf.Size() - aInBufReadPos;
sl@0
  1080
	TInt len = Min(needed, available);
sl@0
  1081
	TPtrC8 ptr8 = aInBuf.Mid(aInBufReadPos, len);
sl@0
  1082
	aInBufReadPos += len;
sl@0
  1083
	
sl@0
  1084
	len >>= K8to16bitShift;
sl@0
  1085
	aOutBuf.Append((const TUint16*)ptr8.Ptr(), len);
sl@0
  1086
	}
sl@0
  1087
sl@0
  1088
/**
sl@0
  1089
Cleans up the allocated during the backup resources - file handles, buffers allocated for the file names.
sl@0
  1090
*/
sl@0
  1091
void CSqlBurCallback::BackupCleanup()
sl@0
  1092
	{
sl@0
  1093
	for(TInt i=0;i<iFileList.Count();++i)
sl@0
  1094
		{
sl@0
  1095
		delete iFileList[i];
sl@0
  1096
		}
sl@0
  1097
	iFileList.Close();
sl@0
  1098
	iFile.Close();
sl@0
  1099
	}
sl@0
  1100
sl@0
  1101
/**
sl@0
  1102
Deletes created during the restore temporary files.
sl@0
  1103
*/
sl@0
  1104
TInt CSqlBurCallback::RestoreCleanup()
sl@0
  1105
	{
sl@0
  1106
	if(iRestoreDir.Find(KSqlBurRestoreDir) < 0)
sl@0
  1107
		{//iRestoreDir is not initialized - that means RestoreCleanup() was called either from the
sl@0
  1108
		 //destructor or from the TerminateMultistageOperation() during a backup.
sl@0
  1109
		return KErrNone;
sl@0
  1110
		}
sl@0
  1111
	iFile.Close();
sl@0
  1112
	CFileMan* fm = NULL;
sl@0
  1113
	TRAPD(err, fm = CFileMan::NewL(iInterface.Fs()));
sl@0
  1114
	if(err == KErrNone)
sl@0
  1115
		{
sl@0
  1116
		TFileName allFiles;
sl@0
  1117
		allFiles.Copy(iRestoreDir);
sl@0
  1118
		allFiles.Append(KSqlBurAllFiles);
sl@0
  1119
		err = fm->Delete(allFiles);
sl@0
  1120
		delete fm;
sl@0
  1121
		}
sl@0
  1122
	return err;
sl@0
  1123
	}
sl@0
  1124
sl@0
  1125
/**
sl@0
  1126
Stores the error occured during backup for furhter processing.
sl@0
  1127
Please note that the function asserts if the aError parameter is KErrNone.
sl@0
  1128
Call the function only with a real error.
sl@0
  1129
sl@0
  1130
@param aError The backup error to be stored 
sl@0
  1131
*/
sl@0
  1132
void CSqlBurCallback::SetBackupError(TInt aError)
sl@0
  1133
	{
sl@0
  1134
	__ASSERT_DEBUG(aError != KErrNone, __SQLPANIC(ESqlPanicBadArgument));
sl@0
  1135
	if(aError != KErrNotFound && aError != KErrPathNotFound)
sl@0
  1136
		{
sl@0
  1137
		if(iBackupError == KErrNone || aError == KErrDiskFull || aError == KErrCorrupt)
sl@0
  1138
			{
sl@0
  1139
			iBackupError = aError;
sl@0
  1140
			}
sl@0
  1141
		}
sl@0
  1142
	}