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".
8 // Initial Contributors:
9 // Nokia Corporation - initial contribution.
17 #include "SqlAssert.h"
18 #include "OstTraceDefinitions.h"
19 #ifdef OST_TRACE_COMPILER_IN_USE
20 #include "SqlBurTraces.h"
22 #include "SqlTraceDef.h"
24 #define UNUSED_ARG(arg) arg = arg
26 _LIT(KSqlBurBackupExt, ".bak");
27 _LIT(KSqlBurRestoreDir, "temprestore");
28 _LIT(KSqlBurAllFiles, "*");
30 const TUint K8to16bitShift = 1;
32 //Extracts and returns 32-bit integer from aNumBuf buffer.
33 static TUint32 GetNumUint32L(const TDesC& aNumBuf)
37 TUint32 num = 0xFFFFFFFF;
38 __SQLLEAVE_IF_ERROR2(lex.Val(num, EHex));
42 //Extracts and returns 64-bit integer from aNumBuf buffer.
43 static TInt64 GetNumInt64L(const TDesC& aNumBuf)
48 __SQLLEAVE_IF_ERROR2(lex.Val(num, EHex));
52 ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////
53 /////////////////////////////// CSqlBurEventMonitor //////////////////////////////////////////////////////////
54 ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////
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.
63 CSqlBurEventMonitor* CSqlBurEventMonitor::NewL(MSqlSrvBurInterface& aInterface)
65 CSqlBurEventMonitor* self = new (ELeave) CSqlBurEventMonitor(aInterface);
66 CleanupStack::PushL(self);
68 CleanupStack::Pop(self);
69 SQL_TRACE_BUR(OstTrace1(TRACE_INTERNALS, CSQLBACKUPNOTIFIER_NEWL, "0x%X;CSqlBurEventMonitor::NewL", (TUint)self));
74 Releases the allocated resources.
76 CSqlBurEventMonitor::~CSqlBurEventMonitor()
78 SQL_TRACE_BUR(OstTrace1(TRACE_INTERNALS, CSQLBACKUPNOTIFIER_CSQLBACKUPNOTIFIER2, "0x%X;CSqlBurEventMonitor::~CSqlBurEventMonitor", (TUint)this));
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.
88 CSqlBurEventMonitor::CSqlBurEventMonitor(MSqlSrvBurInterface& aInterface) :
89 CActive(EPriorityStandard),
90 iBurInterface(aInterface)
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.
99 void CSqlBurEventMonitor::ConstructL()
101 __SQLLEAVE_IF_ERROR(iBurProperty.Attach(KSqlBurPropertyCategoryUid, KSqlBurBackupRestoreKey));
102 CActiveScheduler::Add(this);
103 iBurProperty.Subscribe(iStatus);
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.
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
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.
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
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.
149 @see CSqlBurEventMonitor::RunError()
151 @see CActiveBackupClient
152 @see CSqlBurCallback::InitialiseGetProxyBackupDataL()
153 @see CSqlBurCallback::GetBackupDataSectionL()
154 @see CSqlBurCallback::InitialiseRestoreProxyBaseDataL()
155 @see CSqlBurCallback::RestoreBaseDataSectionL()
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.
160 void CSqlBurEventMonitor::RunL()
162 SQL_TRACE_BUR(OstTrace1(TRACE_INTERNALS, CSQLBACKUPNOTIFIER_RUNL_ENTRY, "Entry;0x%X;CSqlBurEventMonitor::RunL", (TUint)this));
163 iBurProperty.Subscribe(iStatus);
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));
171 SQL_TRACE_BUR(OstTraceExt2(TRACE_INTERNALS, CSQLBACKUPNOTIFIER_RUNL2, "0x%X;CSqlBurEventMonitor::RunL;status=%{TBURPartType}", (TUint)this, status));
175 case conn::EBURBackupFull:
176 case conn::EBURBackupPartial:
177 case conn::EBURRestoreFull:
178 case conn::EBURRestorePartial:
180 // we only do full backups and full restores
181 if(!(iSqlBurCallback && iActiveBackupClient))
184 TRAPD(err, CreateContentL());
191 iActiveBackupClient->ConfirmReadyForBURL(KErrNone);
194 //case conn::EBURUnset:
195 //case conn::EBURNormal:
200 SQL_TRACE_BUR(OstTrace1(TRACE_INTERNALS, CSQLBACKUPNOTIFIER_EXIT, "Exit;0x%X;CSqlBurEventMonitor::RunL", (TUint)this));
205 Cancels the subscribtion for {KUidSystemCategory, KUidBackupRestoreKey} property changes.
207 void CSqlBurEventMonitor::DoCancel()
209 iBurProperty.Cancel();
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.
218 @see CActiveScheduler::Error()
219 @see CSqlBurEventMonitor::RunL()
221 @return The RunL() error, if the RunL() call leaves.
222 @param The RunL() error
224 TInt CSqlBurEventMonitor::RunError(TInt 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);
234 Creates iActiveBackupClient and iSqlBurCallback objects.
236 void CSqlBurEventMonitor::CreateContentL()
238 iSqlBurCallback = CSqlBurCallback::NewL(iBurInterface);
239 iActiveBackupClient = conn::CActiveBackupClient::NewL(iSqlBurCallback);
243 Destroys iActiveBackupClient and iSqlBurCallback objects.
245 void CSqlBurEventMonitor::DestroyContent()
247 delete iActiveBackupClient;
248 iActiveBackupClient = NULL;
249 delete iSqlBurCallback;
250 iSqlBurCallback = NULL;
253 ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////
254 /////////////////////////////// CSqlBackupClient /////////////////////////////////////////////////////////////
255 ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////
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.
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.
271 CSqlBurCallback* CSqlBurCallback::NewL(MSqlSrvBurInterface& aInterface)
273 CSqlBurCallback* self = new (ELeave) CSqlBurCallback(aInterface);
274 SQL_TRACE_BUR(OstTrace1(TRACE_INTERNALS, CSQLBACKUPCLIENT_NEWLC, "0x%X;CSqlBurCallback::NewL", (TUint)self));
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.
282 CSqlBurCallback::CSqlBurCallback(MSqlSrvBurInterface& aInterface) :
283 iInterface(aInterface)
288 Releases the allocated resources.
290 CSqlBurCallback::~CSqlBurCallback()
292 SQL_TRACE_BUR(OstTraceExt2(TRACE_INTERNALS, CSQLBACKUPCLIENT_CSQLBACKUPCLIENT2, "0x%X;CSqlBurCallback::~CSqlBurCallback;iFile.SubSessionHandle()=0x%X", (TUint)this, (TUint)iFile.SubSessionHandle()));
294 (void)RestoreCleanup();
297 ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
298 ///////////////////////////////////// Full backup //////////////////////////////////////////////////////////
299 ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
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.
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.
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.
314 @see MSqlSrvBurInterface
315 @see CSqlBurEventMonitor::RunL()
316 @see CSqlBurCallback::GetBackupDataSectionL()
317 @see CSqlServer::GetBackUpListL()
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.
324 void CSqlBurCallback::InitialiseGetProxyBackupDataL(TSecureId aSid, TDriveNumber aDrive)
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));
328 iInterface.GetBackUpListL(aSid, aDrive, iFileList);
330 iState = EBackupNoFileOpen;
331 iBackupError = KErrNone;
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
344 @see CSqlBurEventMonitor::RunL()
345 @see CSqlBurCallback::InitialiseGetProxyBackupDataL()
347 @param aDrive Unused parameter (the drive number is logged in OST builds).
348 @return an arbitrary number (1024 at the moment)
350 TUint CSqlBurCallback::GetExpectedDataSize(TDriveNumber 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;
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.
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.
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.
386 @see CSqlBurEventMonitor::RunL()
387 @see CSqlBurCallback::InitialiseGetProxyBackupDataL()
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.
394 void CSqlBurCallback::GetBackupDataSectionL(TPtr8& aBuffer, TBool& aFinishedFlag)
396 // don't assume they set it to false
397 aFinishedFlag=EFalse;
398 // any files to backup
399 if(iFileList.Count()==0)
402 SQL_TRACE_BUR(OstTrace1(TRACE_INTERNALS, CSQLBACKUPCLIENT_GETBACKUPDATASECTIONL1, "0x%X;CSqlBurCallback::GetBackupDataSectionL;file count is 0", (TUint)this));
403 aFinishedFlag = ETrue;
408 // run the state machine
409 for(TInt bufFreeSpace=aBuffer.MaxSize()-aBuffer.Size(); bufFreeSpace>0; bufFreeSpace=aBuffer.MaxSize()-aBuffer.Size())
413 case EBackupNoFileOpen: // open a file for processing
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;
420 __SQLLEAVE_IF_ERROR(iBackupError);
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));
429 // there's nothing we can do if we can't open the file so we just skip it
434 iState=EBackupOpenNothingSent;
437 case EBackupOpenNothingSent: // nothing sent (so far) for this file - send the header info
440 TInt err = iFile.Size(fileSize);
444 iState = EBackupEndOfFile;
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));
453 //An error occured while reading the file (or there was not enough memory for the read buffer)
455 iState = EBackupEndOfFile;
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;
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"),
466 KSqlBurMagicNum, // %8x
467 KSqlBurHeaderVersion, // %4x
469 fname.Length(), // %8x
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));
473 // we need it to look like an 8bit buffer
474 TPtr8 hdrPtr8((TUint8*)iBuffer.Ptr(), iBuffer.Size(), iBuffer.Size());
476 TInt len = Min(hdrPtr8.Size(), bufFreeSpace);
478 // append the header to the buffer (only till it's full)
479 aBuffer.Append(hdrPtr8.Ptr(), len);
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)
486 iState = EBackupOpenAllHeaderSent;
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
494 iState = EBackupOpenPartHeaderSent;
498 case EBackupOpenPartHeaderSent: // need to send the rest of the header
500 // get back the header - this is already loaded with the necessary info
501 // from the previous state we were in - EBackupOpenNothingSent
503 // we need it to look like an 8bit buffer
504 TPtr8 hdrPtr8((TUint8*)iBuffer.Ptr(), iBuffer.Size(), iBuffer.Size());
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);
511 if(bytesRemaining <= bufFreeSpace)
513 iHeaderSent = 0; // ready for next header
514 iState = EBackupOpenAllHeaderSent;
518 iHeaderSent += len; // ready to do round again
519 //iState=EBackupOpenPartHeaderSent; same state as now!
523 case EBackupOpenAllHeaderSent: // need to send some data
525 TPtr8 ptr((TUint8*)aBuffer.Ptr() + aBuffer.Size(), 0, bufFreeSpace);
526 TInt err = iFile.Read(ptr);
529 //An error occured while reading the file
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));
535 TInt bytesRead = ptr.Size();
536 aBuffer.SetLength(aBuffer.Size() + bytesRead);
540 iState = EBackupEndOfFile;
545 case EBackupEndOfFile:
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));
549 ++iFileIndex; // move on to next file
550 iState = EBackupNoFileOpen; // go round again
554 __ASSERT_DEBUG(EFalse, __SQLPANIC(ESqlPanicInternalError));
556 }//end of the "switch" statement
557 }//end of the "for" statement
560 ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
561 ///////////////////////////////////// Full restore /////////////////////////////////////////////////////////
562 ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
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.
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.
574 During RestoreBaseDataSectionLL() calls the CSqlBurCallback object will receive data from the backup and resore server.
576 @see CSqlBurEventMonitor::RunL()
577 @see CSqlBurCallback::RestoreBaseDataSectionL()
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.
584 void CSqlBurCallback::InitialiseRestoreProxyBaseDataL(TSecureId aSid, TDriveNumber 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));
590 iState = ERestoreExpectChecksum;
591 iRestoreDrive = aDrive;
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)
605 __SQLLEAVE_IF_ERROR(err);
607 //Cleanup the restore directory
608 err = RestoreCleanup();
609 if(err != KErrNotFound)
611 __SQLLEAVE_IF_ERROR(err);
616 This is called by BUE when the restore has completed.
618 @see CSqlBurEventMonitor::RunL()
619 @see CSqlBurCallback::InitialiseRestoreProxyBaseDataL()
621 @param aDrive the drive that is being restored. Not used (only logged in OST builds).
623 void CSqlBurCallback::RestoreComplete(TDriveNumber 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);
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.
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.
650 @see CSqlBurEventMonitor::RunL()
651 @see CSqlBurCallback::InitialiseRestoreProxyBaseDataL()
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.
658 void CSqlBurCallback::RestoreBaseDataSectionL(TDesC8& aInBuffer, TBool aFinishedFlag)
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;
665 // to mark when the state machine is through
668 // check whether this is an empty restore
669 if(aFinishedFlag && aInBuffer.Size() == 0)
676 // run the state machine
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)
685 // ran out of data in the chunk
686 // so we return and wait for more data to arrive
691 case ERestoreExpectChecksum: // 16 bytes (the header is UTF16 encoded, 8 unicode characters for the checksum)
693 const TInt KCheckSumStrLen = 8;
694 CopyBufData(aInBuffer, inBufferPos, iBuffer, KCheckSumStrLen);
695 if(iBuffer.Length() == KCheckSumStrLen)
697 iChecksum = ::GetNumUint32L(iBuffer);
698 iState = ERestoreExpectOldFileSize;
703 case ERestoreExpectOldFileSize: // 16 bytes (the header is UTF16 encoded, 8 unicode characters for 32-bit old file size)
705 const TInt KOldFileSizeStrLen = 8;
706 CopyBufData(aInBuffer, inBufferPos, iBuffer, KOldFileSizeStrLen);
707 if(iBuffer.Length() == KOldFileSizeStrLen)
709 TUint32 oldFileSize = ::GetNumUint32L(iBuffer);
710 if(oldFileSize == KSqlBurMagicNum)
712 iState = ERestoreExpectVersion;
716 iFileSize = oldFileSize;
717 iState = ERestoreExpectFileNameSize;
723 case ERestoreExpectVersion:
725 const TInt KVersionStrLen = 4;
726 CopyBufData(aInBuffer, inBufferPos, iBuffer, KVersionStrLen);
727 if(iBuffer.Length() == KVersionStrLen)
729 //Ignore the version: ::GetNumUint32L(iBuffer);
730 //At this stage we know that the version is 2+
731 iState = ERestoreExpectFileSize;
736 case ERestoreExpectFileSize:
738 const TInt KFileSizeStrLen = 16;
739 CopyBufData(aInBuffer, inBufferPos, iBuffer, KFileSizeStrLen);
740 if(iBuffer.Length() == KFileSizeStrLen)
742 iFileSize = GetNumInt64L(iBuffer);
743 iState = ERestoreExpectFileNameSize;
746 SQL_TRACE_BUR(OstTraceExt2(TRACE_INTERNALS, CSQLBACKUPCLIENT_RESTOREBASEDATASECTONL1, "0x%X;CSqlBurCallback::RestoreBaseDataSectionL;iFileSize=%lld", (TUint)this, iFileSize));
749 case ERestoreExpectFileNameSize: // the size of the file name to restore
751 const TInt KFileNameLenStrLen = 8;
752 CopyBufData(aInBuffer, inBufferPos, iBuffer, KFileNameLenStrLen);
753 if(iBuffer.Length() == KFileNameLenStrLen)
755 iFileNameSize = GetNumUint32L(iBuffer);
756 iState = ERestoreExpectFileName;
761 case ERestoreExpectFileName: // the name of the file to restore
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)
767 iState = ERestoreExpectData;
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()));
779 case ERestoreExpectData: // now for the data
781 TInt len = Min((aInBuffer.Size() - inBufferPos), iFileSize);
782 TInt err = iFile.Write(aInBuffer.Mid(inBufferPos, len));
785 (void)RestoreCleanup();
792 iState = ERestoreComplete;
796 case ERestoreComplete: // file completely restored
798 TUint64 checkSum64 = 0;
799 TInt restoreErr = iFile.Flush();
800 if(restoreErr == KErrNone)
802 // calculate the checksum
803 restoreErr = CheckSum(iFile, checkSum64);
806 if(restoreErr != KErrNone)
808 (void)RestoreCleanup();
809 __SQLLEAVE(restoreErr);
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)
815 (void)RestoreCleanup();
816 __SQLLEAVE(KErrCorrupt);
818 if((aInBuffer.Size() - inBufferPos) > 0)
819 {//There are bytes to be consumed in the input buffer
820 iState = ERestoreExpectChecksum;
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));
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)
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)
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)
864 err = iInterface.Fs().Rename(iParse.FullName(), dbName);
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);
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));
876 if(err != KErrNone && err != KErrNotFound)
878 if(restoreErr == KErrNone)
885 }//iInterface.Fs().GetDir(...)
887 (void)RestoreCleanup();
888 if(restoreErr != KErrNone)
890 __SQLLEAVE(restoreErr);
895 __ASSERT_DEBUG(EFalse, __SQLPANIC(ESqlPanicInternalError));
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);
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.
915 void CSqlBurCallback::TerminateMultiStageOperation()
918 (void)RestoreCleanup();
922 We do our own checksumming so we don't need this
924 @param aDrive the drive affected (unused)
926 TUint CSqlBurCallback::GetDataChecksum(TDriveNumber /* aDrive */)
928 // not required - not implemented
929 const TUint KArbitraryNumber = 1024;
930 return KArbitraryNumber;
933 ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
934 /////////////////////////// Incremental backup/restore ////////////////////////////////////////////////////
935 ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
938 We don't support incremental backup
940 void CSqlBurCallback::GetSnapshotDataL(TDriveNumber /* aDrive */, TPtr8& /* aBuffer */, TBool& /* aFinishedFlag */)
942 __SQLLEAVE(KErrNotSupported);
946 We don't support incremental backup
948 void CSqlBurCallback::InitialiseGetBackupDataL(TDriveNumber /* aDrive */)
950 __SQLLEAVE(KErrNotSupported);
954 We don't support incremental backup
956 void CSqlBurCallback::InitialiseRestoreBaseDataL(TDriveNumber /* aDrive */)
958 __SQLLEAVE(KErrNotSupported);
962 We don't support incremental backup
964 void CSqlBurCallback::InitialiseRestoreIncrementDataL(TDriveNumber /* aDrive */)
966 __SQLLEAVE(KErrNotSupported);
970 We don't support incremental backup
972 void CSqlBurCallback::RestoreIncrementDataSectionL(TDesC8& /* aBuffer */, TBool /* aFinishedFlag */)
974 __SQLLEAVE(KErrNotSupported);
978 We don't support incremental backup
980 void CSqlBurCallback::AllSnapshotsSuppliedL()
985 We don't support incremental backup
987 void CSqlBurCallback::ReceiveSnapshotDataL(TDriveNumber /* aDrive */, TDesC8& /* aBuffer */, TBool /* aFinishedFlag */)
989 __SQLLEAVE(KErrNotSupported);
992 ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
993 /////////////////////////// Helper functions //////////////////////////////////////////////////////////////
994 ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
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.
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.
1007 TInt CSqlBurCallback::CheckSum(const RFile64& aOpenFile, TUint64& aCheckSum) const
1009 // scoot through the database file building the checksum
1011 TInt64 seekPos = 0; // rewind first
1012 TInt err = aOpenFile.Seek(ESeekStart, seekPos);
1017 const TUint KCheckSumBlockSize = 4 * 1024;
1018 HBufC8* buf = HBufC8::New(KCheckSumBlockSize);
1021 return KErrNoMemory;
1023 TPtr8 ptr = buf->Des();
1026 err = aOpenFile.Read(ptr);
1032 TInt len = ptr.Length();
1037 // calculate the checksum
1038 for(TInt i=0;i<len;++i)
1040 aCheckSum = (aCheckSum << 1) | (aCheckSum >> 63);
1041 aCheckSum += ptr[i];
1045 // restore file position
1047 err = aOpenFile.Seek(ESeekStart,seekPos);
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.
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).
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.
1073 void CSqlBurCallback::CopyBufData(const TDesC8& aInBuf, TInt& aInBufReadPos, TDes& aOutBuf, TInt aDataLen)
1075 __ASSERT_DEBUG(aInBufReadPos >= 0, __SQLPANIC(ESqlPanicBadArgument));
1076 __ASSERT_DEBUG(aDataLen > 0, __SQLPANIC(ESqlPanicBadArgument));
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;
1084 len >>= K8to16bitShift;
1085 aOutBuf.Append((const TUint16*)ptr8.Ptr(), len);
1089 Cleans up the allocated during the backup resources - file handles, buffers allocated for the file names.
1091 void CSqlBurCallback::BackupCleanup()
1093 for(TInt i=0;i<iFileList.Count();++i)
1095 delete iFileList[i];
1102 Deletes created during the restore temporary files.
1104 TInt CSqlBurCallback::RestoreCleanup()
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.
1112 CFileMan* fm = NULL;
1113 TRAPD(err, fm = CFileMan::NewL(iInterface.Fs()));
1117 allFiles.Copy(iRestoreDir);
1118 allFiles.Append(KSqlBurAllFiles);
1119 err = fm->Delete(allFiles);
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.
1130 @param aError The backup error to be stored
1132 void CSqlBurCallback::SetBackupError(TInt aError)
1134 __ASSERT_DEBUG(aError != KErrNone, __SQLPANIC(ESqlPanicBadArgument));
1135 if(aError != KErrNotFound && aError != KErrPathNotFound)
1137 if(iBackupError == KErrNone || aError == KErrDiskFull || aError == KErrCorrupt)
1139 iBackupError = aError;