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