sl@0: // Copyright (c) 2005-2010 Nokia Corporation and/or its subsidiary(-ies). sl@0: // All rights reserved. sl@0: // This component and the accompanying materials are made available sl@0: // under the terms of "Eclipse Public License v1.0" sl@0: // which accompanies this distribution, and is available sl@0: // at the URL "http://www.eclipse.org/legal/epl-v10.html". sl@0: // sl@0: // Initial Contributors: sl@0: // Nokia Corporation - initial contribution. sl@0: // sl@0: // Contributors: sl@0: // sl@0: // Description: sl@0: // sl@0: sl@0: #include "SqlBur.h" sl@0: #include "SqlAssert.h" sl@0: #include "OstTraceDefinitions.h" sl@0: #ifdef OST_TRACE_COMPILER_IN_USE sl@0: #include "SqlBurTraces.h" sl@0: #endif sl@0: #include "SqlTraceDef.h" sl@0: sl@0: #define UNUSED_ARG(arg) arg = arg sl@0: sl@0: _LIT(KSqlBurBackupExt, ".bak"); sl@0: _LIT(KSqlBurRestoreDir, "temprestore"); sl@0: _LIT(KSqlBurAllFiles, "*"); sl@0: sl@0: const TUint K8to16bitShift = 1; sl@0: sl@0: //Extracts and returns 32-bit integer from aNumBuf buffer. sl@0: static TUint32 GetNumUint32L(const TDesC& aNumBuf) sl@0: { sl@0: TLex lex(aNumBuf); sl@0: lex.SkipSpace(); sl@0: TUint32 num = 0xFFFFFFFF; sl@0: __SQLLEAVE_IF_ERROR2(lex.Val(num, EHex)); sl@0: return num; sl@0: } sl@0: sl@0: //Extracts and returns 64-bit integer from aNumBuf buffer. sl@0: static TInt64 GetNumInt64L(const TDesC& aNumBuf) sl@0: { sl@0: TLex lex(aNumBuf); sl@0: lex.SkipSpace(); sl@0: TInt64 num = -1; sl@0: __SQLLEAVE_IF_ERROR2(lex.Val(num, EHex)); sl@0: return num; sl@0: } sl@0: sl@0: /////////////////////////////////////////////////////////////////////////////////////////////////////////////////// sl@0: /////////////////////////////// CSqlBurEventMonitor ////////////////////////////////////////////////////////// sl@0: /////////////////////////////////////////////////////////////////////////////////////////////////////////////////// sl@0: sl@0: /** sl@0: Standard two phase construction. Creates a CSqlBurEventMonitor instance. sl@0: @param aInterface A reference to an interface that is used for retrieving list of databases to be sent for backup. sl@0: @return An instance of the backup notifier sl@0: @leave KErrNoMemory, an out of memory condition has occurred; sl@0: Note that the function may also leave with some other system-wide error codes. sl@0: */ sl@0: CSqlBurEventMonitor* CSqlBurEventMonitor::NewL(MSqlSrvBurInterface& aInterface) sl@0: { sl@0: CSqlBurEventMonitor* self = new (ELeave) CSqlBurEventMonitor(aInterface); sl@0: CleanupStack::PushL(self); sl@0: self->ConstructL(); sl@0: CleanupStack::Pop(self); sl@0: SQL_TRACE_BUR(OstTrace1(TRACE_INTERNALS, CSQLBACKUPNOTIFIER_NEWL, "0x%X;CSqlBurEventMonitor::NewL", (TUint)self)); sl@0: return self; sl@0: } sl@0: sl@0: /** sl@0: Releases the allocated resources. sl@0: */ sl@0: CSqlBurEventMonitor::~CSqlBurEventMonitor() sl@0: { sl@0: SQL_TRACE_BUR(OstTrace1(TRACE_INTERNALS, CSQLBACKUPNOTIFIER_CSQLBACKUPNOTIFIER2, "0x%X;CSqlBurEventMonitor::~CSqlBurEventMonitor", (TUint)this)); sl@0: Cancel(); sl@0: iBurProperty.Close(); sl@0: DestroyContent(); sl@0: } sl@0: sl@0: /** sl@0: Initializes data members with their default values. sl@0: @param aInterface A reference to an interface that is used for retrieving list of databases to be sent for backup. sl@0: */ sl@0: CSqlBurEventMonitor::CSqlBurEventMonitor(MSqlSrvBurInterface& aInterface) : sl@0: CActive(EPriorityStandard), sl@0: iBurInterface(aInterface) sl@0: { sl@0: } sl@0: sl@0: /** sl@0: Initializes the created CSqlBurEventMonitor object. sl@0: @leave KErrNoMemory, an out of memory condition has occurred; sl@0: Note that the function may also leave with some other system-wide error codes. sl@0: */ sl@0: void CSqlBurEventMonitor::ConstructL() sl@0: { sl@0: __SQLLEAVE_IF_ERROR(iBurProperty.Attach(KSqlBurPropertyCategoryUid, KSqlBurBackupRestoreKey)); sl@0: CActiveScheduler::Add(this); sl@0: iBurProperty.Subscribe(iStatus); sl@0: SetActive(); sl@0: } sl@0: sl@0: /** sl@0: RunL() is called when the value of the {KUidSystemCategory, KUidBackupRestoreKey} gets changed. sl@0: That indicates: a backup or a restore is about to begin. sl@0: sl@0: How the function works: sl@0: - When a backup or restore notification is received, the function will subscribe again for notifications from sl@0: the backup and restore property and will read the property status; sl@0: - If the property status is conn::EBURUnset or conn::EBURNormal, the function will destroy iSqlBurCallback sl@0: and iActiveBackupClient interfaces. No more callbacks will be reseived from the backup and restore server. sl@0: This is the end of the backup or restore processing; sl@0: - If the property status is conn::EBURBackupFull, conn::EBURBackupPartial, conn::EBURRestoreFull or sl@0: conn::EBURRestorePartial, the function will create iSqlBurCallback and iActiveBackupClient interface sl@0: (iActiveBackupClient's NewL() receives iSqlBurCallback as an input parameter, registering this way the callback sl@0: in the backup and restore server to be called later, when sending or retrieving data to/from the server). sl@0: If the property read and the interface creation operations have been successful, the function will call sl@0: ConfirmReadyForBURL(KErrNone) to notify the backup and restore server that the SQL server is ready to send/retrieve sl@0: backup/restore data. sl@0: If the current notification is that a backup is about to begin, after the confirmation the backup and restore server will sl@0: call CSqlBurCallback::InitialiseGetProxyBackupDataL() once per {client secure id, drive} sl@0: followed by CSqlBurCallback::GetBackupDataSectionL() calls to retrieve the backup data. sl@0: If the current notification is that a restore is about to begin, after the confirmation the backup and restore server will sl@0: call CSqlBurCallback::InitialiseRestoreProxyBaseDataL() once per {client secure id, drive} sl@0: followed by CSqlBurCallback::RestoreBaseDataSectionL() calls to send the restore data. sl@0: sl@0: The current implementation has one design flaw. If a backup or restore notification is received, there are at lest 3 sl@0: places before the ConfirmReadyForBURL() call, where the code may leave: sl@0: - the "property get" operation; sl@0: - the iSqlBurCallback creation; sl@0: - the iActiveBackupClient creation; sl@0: If a leave occurs at some of the mentioned places, that leave will be trapped by the current CActiveScheduler object sl@0: and CSqlBurEventMonitor::RunError() will be called with the leaved error code. sl@0: Problem #1: CSqlBurEventMonitor::RunError() won't do anything with the error (apart from printing a trace in the OST builds). sl@0: The error is silently suppressed. The backup or restore won't start. But the client won't see any notification sl@0: for that problem. sl@0: Problem #2: ConfirmReadyForBURL() won't be called. According to the backup and restore documentation, if sl@0: ConfirmReadyForBURL() is called with KErrNone parameter, that's a confirmation for the backup and restore sl@0: server to start the processing. If ConfirmReadyForBURL() is called with an error different than KErrNone, sl@0: that's a confirmation for the backup and restore server that the client is not ready. No backup or restore sl@0: will be started. The remote backup client will be notified about the problem. sl@0: After an investigation it was found that the same problems do exist in all active backup clients, none of them has sl@0: solved the problems. Then, the code here will be kept as it is, it might be too dangerous to do a change right now. sl@0: sl@0: @see CSqlBurEventMonitor::RunError() sl@0: @see CSqlBurCallback sl@0: @see CActiveBackupClient sl@0: @see CSqlBurCallback::InitialiseGetProxyBackupDataL() sl@0: @see CSqlBurCallback::GetBackupDataSectionL() sl@0: @see CSqlBurCallback::InitialiseRestoreProxyBaseDataL() sl@0: @see CSqlBurCallback::RestoreBaseDataSectionL() sl@0: sl@0: @leave KErrNoMemory, an out of memory condition has occurred; sl@0: Note that the function may also leave with some other system-wide error codes. sl@0: */ sl@0: void CSqlBurEventMonitor::RunL() sl@0: { sl@0: SQL_TRACE_BUR(OstTrace1(TRACE_INTERNALS, CSQLBACKUPNOTIFIER_RUNL_ENTRY, "Entry;0x%X;CSqlBurEventMonitor::RunL", (TUint)this)); sl@0: iBurProperty.Subscribe(iStatus); sl@0: SetActive(); sl@0: TInt status; sl@0: __SQLLEAVE_IF_ERROR(iBurProperty.Get(status)); sl@0: status &= conn::KBURPartTypeMask; sl@0: #ifdef _SQL_RDEBUG_PRINT sl@0: SQL_TRACE_BUR(OstTraceExt2(TRACE_INTERNALS, CSQLBACKUPNOTIFIER_RUNL1, "0x%X;CSqlBurEventMonitor::RunL;status=%d", (TUint)this, status)); sl@0: #else sl@0: SQL_TRACE_BUR(OstTraceExt2(TRACE_INTERNALS, CSQLBACKUPNOTIFIER_RUNL2, "0x%X;CSqlBurEventMonitor::RunL;status=%{TBURPartType}", (TUint)this, status)); sl@0: #endif sl@0: switch(status) sl@0: { sl@0: case conn::EBURBackupFull: sl@0: case conn::EBURBackupPartial: sl@0: case conn::EBURRestoreFull: sl@0: case conn::EBURRestorePartial: sl@0: { sl@0: // we only do full backups and full restores sl@0: if(!(iSqlBurCallback && iActiveBackupClient)) sl@0: { sl@0: DestroyContent(); sl@0: TRAPD(err, CreateContentL()); sl@0: if(err != KErrNone) sl@0: { sl@0: DestroyContent(); sl@0: __SQLLEAVE(err); sl@0: } sl@0: } sl@0: iActiveBackupClient->ConfirmReadyForBURL(KErrNone); sl@0: } sl@0: break; sl@0: //case conn::EBURUnset: sl@0: //case conn::EBURNormal: sl@0: default: sl@0: DestroyContent(); sl@0: break; sl@0: } sl@0: SQL_TRACE_BUR(OstTrace1(TRACE_INTERNALS, CSQLBACKUPNOTIFIER_EXIT, "Exit;0x%X;CSqlBurEventMonitor::RunL", (TUint)this)); sl@0: SQL_BUR_TEST_STOP(); sl@0: } sl@0: sl@0: /** sl@0: Cancels the subscribtion for {KUidSystemCategory, KUidBackupRestoreKey} property changes. sl@0: */ sl@0: void CSqlBurEventMonitor::DoCancel() sl@0: { sl@0: iBurProperty.Cancel(); sl@0: } sl@0: sl@0: /** sl@0: No-op. The method does nothing with the reported from CSqlBurEventMonitor::RunL() error sl@0: (apart from logging a trace in OST builds). sl@0: Actually, the right action is to return KErrNone (as it is implemented), otherwise the default implementation of sl@0: CActiveScheduler::Error() will panic the current thread. sl@0: sl@0: @see CActiveScheduler::Error() sl@0: @see CSqlBurEventMonitor::RunL() sl@0: sl@0: @return The RunL() error, if the RunL() call leaves. sl@0: @param The RunL() error sl@0: */ sl@0: TInt CSqlBurEventMonitor::RunError(TInt aError) sl@0: { sl@0: UNUSED_ARG(aError); sl@0: SQL_TRACE_BUR(OstTraceExt2(TRACE_INTERNALS, CSQLBACKUPCLIENT_RUNERROR, "0x%X;CSqlBurEventMonitor::RunError;aError=%d", (TUint)this, aError)); sl@0: SQL_BUR_TEST_SET_ERROR(aError); sl@0: SQL_BUR_TEST_STOP(); sl@0: return KErrNone; sl@0: } sl@0: sl@0: /** sl@0: Creates iActiveBackupClient and iSqlBurCallback objects. sl@0: */ sl@0: void CSqlBurEventMonitor::CreateContentL() sl@0: { sl@0: iSqlBurCallback = CSqlBurCallback::NewL(iBurInterface); sl@0: iActiveBackupClient = conn::CActiveBackupClient::NewL(iSqlBurCallback); sl@0: } sl@0: sl@0: /** sl@0: Destroys iActiveBackupClient and iSqlBurCallback objects. sl@0: */ sl@0: void CSqlBurEventMonitor::DestroyContent() sl@0: { sl@0: delete iActiveBackupClient; sl@0: iActiveBackupClient = NULL; sl@0: delete iSqlBurCallback; sl@0: iSqlBurCallback = NULL; sl@0: } sl@0: sl@0: /////////////////////////////////////////////////////////////////////////////////////////////////////////////////// sl@0: /////////////////////////////// CSqlBackupClient ///////////////////////////////////////////////////////////// sl@0: /////////////////////////////////////////////////////////////////////////////////////////////////////////////////// sl@0: sl@0: sl@0: /** sl@0: Creates new CSqlBurCallback instance. sl@0: The CSqlBurEventMonitor object monitors the state of the {KUidSystemCategory, KUidBackupRestoreKey} sl@0: property. When a backup or a restore is about to begin, the CSqlBurEventMonitor object creates a sl@0: CSqlBurCallback instance, establishes a connection with the B&R server and passes a pointer to sl@0: the CSqlBurCallback callback to the BYR conenction. sl@0: The CSqlBurCallback methods will be called during the backup/restore for sending/retrieving data. sl@0: sl@0: @param aInterface A reference to an interface that is used for retrieving list of databases to be sent for backup. sl@0: @return A pointer to the created CSqlBurCallback instance sl@0: @leave KErrNoMemory, an out of memory condition has occurred; sl@0: Note that the function may also leave with some other system-wide error codes. sl@0: */ sl@0: CSqlBurCallback* CSqlBurCallback::NewL(MSqlSrvBurInterface& aInterface) sl@0: { sl@0: CSqlBurCallback* self = new (ELeave) CSqlBurCallback(aInterface); sl@0: SQL_TRACE_BUR(OstTrace1(TRACE_INTERNALS, CSQLBACKUPCLIENT_NEWLC, "0x%X;CSqlBurCallback::NewL", (TUint)self)); sl@0: return self; sl@0: } sl@0: sl@0: /** sl@0: Initializes CSqlBurCallback data members with their default values. sl@0: @param aInterface A reference to an interface that is used for retrieving list of databases to be sent for backup. sl@0: */ sl@0: CSqlBurCallback::CSqlBurCallback(MSqlSrvBurInterface& aInterface) : sl@0: iInterface(aInterface) sl@0: { sl@0: } sl@0: sl@0: /** sl@0: Releases the allocated resources. sl@0: */ sl@0: CSqlBurCallback::~CSqlBurCallback() sl@0: { sl@0: SQL_TRACE_BUR(OstTraceExt2(TRACE_INTERNALS, CSQLBACKUPCLIENT_CSQLBACKUPCLIENT2, "0x%X;CSqlBurCallback::~CSqlBurCallback;iFile.SubSessionHandle()=0x%X", (TUint)this, (TUint)iFile.SubSessionHandle())); sl@0: BackupCleanup(); sl@0: (void)RestoreCleanup(); sl@0: } sl@0: sl@0: //////////////////////////////////////////////////////////////////////////////////////////////////////////////////// sl@0: ///////////////////////////////////// Full backup ////////////////////////////////////////////////////////// sl@0: //////////////////////////////////////////////////////////////////////////////////////////////////////////////////// sl@0: sl@0: /** sl@0: This is called to let us know that the given SID is to be backed up. sl@0: We ask the SQL server (using iInterface, see MSqlSrvBurInterface for more details) sl@0: for a list of databases that want to be backed up. sl@0: sl@0: The backup is initiated by a notification received in CSqlBurEventMonitor::RunL() method. sl@0: InitialiseGetProxyBackupDataL() is called once per {client secure id, drive} and each sl@0: InitialiseGetProxyBackupDataL() call is followed after that by a set of CSqlBurCallback::GetBackupDataSectionL() calls, sl@0: made from the backup and restore client dll. sl@0: sl@0: During GetBackupDataSectionL() calls the CSqlBurCallback object will read the content of the databases from the list, sl@0: retrieved from the MSqlSrvBurInterface::GetBackUpListL() call and send the content to the backup and restore server. sl@0: sl@0: @see MSqlSrvBurInterface sl@0: @see CSqlBurEventMonitor::RunL() sl@0: @see CSqlBurCallback::GetBackupDataSectionL() sl@0: @see CSqlServer::GetBackUpListL() sl@0: sl@0: @param aSid the UID of the application to backup sl@0: @param aDrive the drive to be backed up sl@0: @leave KErrNoMemory, an out of memory condition has occurred; sl@0: Note that the function may also leave with some other system-wide error codes. sl@0: */ sl@0: void CSqlBurCallback::InitialiseGetProxyBackupDataL(TSecureId aSid, TDriveNumber aDrive) sl@0: { sl@0: SQL_TRACE_BUR(OstTraceExt3(TRACE_INTERNALS, CSQLBACKUPCLIENT_INITIALIZEGETPROXYBACKUPDATAL, "0x%X;CSqlBurCallback::InitialiseGetProxyBackupDataL;aSid=0x%X;aDrive=%d", (TUint)this, (TUint)aSid.iId, (TInt)aDrive)); sl@0: BackupCleanup(); sl@0: iInterface.GetBackUpListL(aSid, aDrive, iFileList); sl@0: iFileIndex = 0; sl@0: iState = EBackupNoFileOpen; sl@0: iBackupError = KErrNone; sl@0: } sl@0: sl@0: /** sl@0: This is supposed to allow the B&R framework to know in advance how much sl@0: data is coming - but unfortunately there is no way to know this sl@0: at this stage since we don't even know yet what SID is being processed sl@0: So we just answer some number to make the BUE happy. It doesn't sl@0: actually rely on this number so there is no risk - the aFinishedFlag sl@0: indicates the end of data, not the value returned here. It is sl@0: supposed to allow the BUE to optimise its behaviour by know up front sl@0: the data volume. sl@0: sl@0: @see CSqlBurEventMonitor::RunL() sl@0: @see CSqlBurCallback::InitialiseGetProxyBackupDataL() sl@0: sl@0: @param aDrive Unused parameter (the drive number is logged in OST builds). sl@0: @return an arbitrary number (1024 at the moment) sl@0: */ sl@0: TUint CSqlBurCallback::GetExpectedDataSize(TDriveNumber aDrive) sl@0: { sl@0: UNUSED_ARG(aDrive); sl@0: // we have no idea at this point - we even don't know who is to be backed up yet sl@0: const TUint KArbitraryNumber = 1024; sl@0: SQL_TRACE_BUR(OstTraceExt3(TRACE_INTERNALS, CSQLBACKUPCLIENT_GETEXPECTEDDATASIZE, "0x%X;CSqlBurCallback::GetExpectedDataSize;aDrive=%d;rc=%u", (TUint)this, (TInt)aDrive, KArbitraryNumber)); sl@0: return KArbitraryNumber; sl@0: } sl@0: sl@0: /** sl@0: This is the backup state machine sl@0: Because the data has to be sent back in sections and the various sl@0: components of the dataflow may straddle chunks, we have to keep sl@0: track of where we are between each transfer - a state machine is sl@0: the simplest and most understandable implementation. sl@0: sl@0: Please note how the function processes the errors occuring during the backup. sl@0: If an error occurs, the error is not propagated back to the B&R server immediatelly. sl@0: The error is stored in iBurError data member and is reported at the end of the backup process. sl@0: The reason for such unusual error reporting poicy is: the SQL server performs full backup of possibly more sl@0: than one database file. If an error occurs during the backup of the first file for example, the backup sl@0: process should not stop at that point. All files will be processed and then at the end, the error will be reproted. sl@0: sl@0: In details, the function runs a state machine, where: sl@0: - every file in the list retrieved in InitialiseGetProxyBackupDataL() is opened; sl@0: - the file is read and 32-bit checksum over the file data - calculated; sl@0: - a file backup header is prepared, including there the file size, file name, file name length, protocol verison number sl@0: and the checksum. The header is sent to the backup restore server; sl@0: - the file data is read and sent to the backup and restore server; sl@0: - during the described above sequence no leave ever occurs. The error that occurs during the file processing, sl@0: is stored into a data member of CSqlBurCallback class. At the end, after the last file in the list is processed, sl@0: the backup and restore server will get a notification (via a User::Leave() call) regarding the error; sl@0: The used error reporting policy allows all files to be process without interrupting the backup process. sl@0: For example, if there are 5 files to be sent to the backup and restore server, an error that occurs during the sl@0: processing of file #3, won't prevent files #4 and #5 from being sent for backup. sl@0: sl@0: @see CSqlBurEventMonitor::RunL() sl@0: @see CSqlBurCallback::InitialiseGetProxyBackupDataL() sl@0: sl@0: @param aBuffer Output parameter, the buffer where the data will be put to be passed back sl@0: @param aFinishedFlag Set to true when all data has been submitted for backup sl@0: @leave KErrNoMemory, an out of memory condition has occurred; sl@0: Note that the function may also leave with some other system-wide error codes. sl@0: */ sl@0: void CSqlBurCallback::GetBackupDataSectionL(TPtr8& aBuffer, TBool& aFinishedFlag) sl@0: { sl@0: // don't assume they set it to false sl@0: aFinishedFlag=EFalse; sl@0: // any files to backup sl@0: if(iFileList.Count()==0) sl@0: { sl@0: // nothing to backup sl@0: SQL_TRACE_BUR(OstTrace1(TRACE_INTERNALS, CSQLBACKUPCLIENT_GETBACKUPDATASECTIONL1, "0x%X;CSqlBurCallback::GetBackupDataSectionL;file count is 0", (TUint)this)); sl@0: aFinishedFlag = ETrue; sl@0: BackupCleanup(); sl@0: return; sl@0: } sl@0: sl@0: // run the state machine sl@0: for(TInt bufFreeSpace=aBuffer.MaxSize()-aBuffer.Size(); bufFreeSpace>0; bufFreeSpace=aBuffer.MaxSize()-aBuffer.Size()) sl@0: { sl@0: switch(iState) sl@0: { sl@0: case EBackupNoFileOpen: // open a file for processing sl@0: { sl@0: if(iFileIndex>=iFileList.Count()) sl@0: {// all files have been processed - send the finished flag sl@0: SQL_TRACE_BUR(OstTraceExt2(TRACE_INTERNALS, CSQLBACKUPCLIENT_GETBACKUPDATASECTIONL2, "0x%X;CSqlBurCallback::GetBackupDataSectionL;all files processed;iBackupError=%d", (TUint)this, iBackupError)); sl@0: aFinishedFlag = ETrue; sl@0: BackupCleanup(); sl@0: __SQLLEAVE_IF_ERROR(iBackupError); sl@0: return; sl@0: } sl@0: // open the database file to send sl@0: TPtrC fname = iFileList[iFileIndex]->Des(); sl@0: TInt err = iFile.Open(iInterface.Fs(), fname, EFileRead | EFileShareExclusive); sl@0: SQL_TRACE_BUR(OstTraceExt5(TRACE_INTERNALS, CSQLBACKUPCLIENT_GETBACKUPDATASECTIONL3, "0x%X;CSqlBurCallback::GetBackupDataSectionL;BEGIN;fname=%S;iFileIndex=%d;iFile.SubSessionHandle()=0x%X;err=%d", (TUint)this, __SQLPRNSTR(fname), iFileIndex, (TUint)iFile.SubSessionHandle(), err)); sl@0: if(KErrNone != err) sl@0: { sl@0: // there's nothing we can do if we can't open the file so we just skip it sl@0: SetBackupError(err); sl@0: ++iFileIndex; sl@0: break; sl@0: } sl@0: iState=EBackupOpenNothingSent; sl@0: break; sl@0: } sl@0: case EBackupOpenNothingSent: // nothing sent (so far) for this file - send the header info sl@0: { sl@0: TInt64 fileSize; sl@0: TInt err = iFile.Size(fileSize); sl@0: if(KErrNone != err) sl@0: { sl@0: SetBackupError(err); sl@0: iState = EBackupEndOfFile; sl@0: break; sl@0: } sl@0: sl@0: TUint64 checksum64 = 0; sl@0: err = CheckSum(iFile, checksum64); sl@0: SQL_TRACE_BUR(OstTraceExt3(TRACE_INTERNALS, CSQLBACKUPCLIENT_GETBACKUPDATASECTIONL4, "0x%X;CSqlBurCallback::GetBackupDataSectionL;CheckSum();iFileIndex=%d;err=%d", (TUint)this, iFileIndex, err)); sl@0: if(err != KErrNone) sl@0: { sl@0: //An error occured while reading the file (or there was not enough memory for the read buffer) sl@0: SetBackupError(err); sl@0: iState = EBackupEndOfFile; sl@0: break; sl@0: } sl@0: // Only grab last 4 bytes of the checksum - enough to be satisfied that the backup and restore worked ok sl@0: TUint32 checksum32 = checksum64 & KMaxTUint32; sl@0: sl@0: // build the header - this is an instance member because it sl@0: // has to persist over multiple calls to this method sl@0: TPtrC fname = iFileList[iFileIndex]->Des(); sl@0: iBuffer.Format(_L("%8x%8x%4x%16lx%8x%S"), sl@0: checksum32, // %8x sl@0: KSqlBurMagicNum, // %8x sl@0: KSqlBurHeaderVersion, // %4x sl@0: fileSize, // %16lx sl@0: fname.Length(), // %8x sl@0: &fname); // %S sl@0: SQL_TRACE_BUR(OstTraceExt4(TRACE_INTERNALS, CSQLBACKUPCLIENT_GETBACKUPDATASECTIONL5, "0x%X;CSqlBackupClient::GetBackupDataSectionL;fileName=%S;hdrPtr=|%S|;fileSize=%lld", (TUint)this, __SQLPRNSTR(fname), __SQLPRNSTR(iBuffer), fileSize)); sl@0: sl@0: // we need it to look like an 8bit buffer sl@0: TPtr8 hdrPtr8((TUint8*)iBuffer.Ptr(), iBuffer.Size(), iBuffer.Size()); sl@0: sl@0: TInt len = Min(hdrPtr8.Size(), bufFreeSpace); sl@0: sl@0: // append the header to the buffer (only till it's full) sl@0: aBuffer.Append(hdrPtr8.Ptr(), len); sl@0: sl@0: // decide what needs to happen next sl@0: // if complete then we need data, otherwise we need to put sl@0: // the rest of the header in the next chunk sl@0: if(hdrPtr8.Size() <= bufFreeSpace) sl@0: { sl@0: iState = EBackupOpenAllHeaderSent; sl@0: } sl@0: else sl@0: { sl@0: // we need to keep track of how much of the header has sl@0: // been sent so that we only send the reminder on the next sl@0: // iteration sl@0: iHeaderSent = len; sl@0: iState = EBackupOpenPartHeaderSent; sl@0: } sl@0: break; sl@0: } sl@0: case EBackupOpenPartHeaderSent: // need to send the rest of the header sl@0: { sl@0: // get back the header - this is already loaded with the necessary info sl@0: // from the previous state we were in - EBackupOpenNothingSent sl@0: sl@0: // we need it to look like an 8bit buffer sl@0: TPtr8 hdrPtr8((TUint8*)iBuffer.Ptr(), iBuffer.Size(), iBuffer.Size()); sl@0: sl@0: // how many bytes have we yet to send? sl@0: TInt bytesRemaining = hdrPtr8.Size() - iHeaderSent; sl@0: TInt len = Min(bytesRemaining, bufFreeSpace); sl@0: aBuffer.Append(hdrPtr8.Ptr() + iHeaderSent, len); sl@0: sl@0: if(bytesRemaining <= bufFreeSpace) sl@0: { sl@0: iHeaderSent = 0; // ready for next header sl@0: iState = EBackupOpenAllHeaderSent; sl@0: } sl@0: else sl@0: { sl@0: iHeaderSent += len; // ready to do round again sl@0: //iState=EBackupOpenPartHeaderSent; same state as now! sl@0: } sl@0: break; sl@0: } sl@0: case EBackupOpenAllHeaderSent: // need to send some data sl@0: { sl@0: TPtr8 ptr((TUint8*)aBuffer.Ptr() + aBuffer.Size(), 0, bufFreeSpace); sl@0: TInt err = iFile.Read(ptr); sl@0: if(err != KErrNone) sl@0: { sl@0: //An error occured while reading the file sl@0: SetBackupError(err); sl@0: iState = EBackupEndOfFile; sl@0: SQL_TRACE_BUR(OstTraceExt3(TRACE_INTERNALS, CSQLBACKUPCLIENT_GETBACKUPDATASECTIONL6, "0x%X;CSqlBurCallback::GetBackupDataSectionL;File read;iFileIndex=%d;err=%d", (TUint)this, iFileIndex, err)); sl@0: break; sl@0: } sl@0: TInt bytesRead = ptr.Size(); sl@0: aBuffer.SetLength(aBuffer.Size() + bytesRead); sl@0: // EOF sl@0: if(bytesRead == 0) sl@0: { sl@0: iState = EBackupEndOfFile; sl@0: break; sl@0: } sl@0: break; sl@0: } sl@0: case EBackupEndOfFile: sl@0: { sl@0: SQL_TRACE_BUR(OstTraceExt3(TRACE_INTERNALS, CSQLBACKUPCLIENT_GETBACKUPDATASECTIONL7, "0x%X;CSqlBurCallback::GetBackupDataSectionL;END;iFile.SubSessionHandle()=0x%X;iFileIndex=%d", (TUint)this, (TUint)iFile.SubSessionHandle(), iFileIndex)); sl@0: iFile.Close(); sl@0: ++iFileIndex; // move on to next file sl@0: iState = EBackupNoFileOpen; // go round again sl@0: break; sl@0: } sl@0: default: sl@0: __ASSERT_DEBUG(EFalse, __SQLPANIC(ESqlPanicInternalError)); sl@0: break; sl@0: }//end of the "switch" statement sl@0: }//end of the "for" statement sl@0: } sl@0: sl@0: //////////////////////////////////////////////////////////////////////////////////////////////////////////////////// sl@0: ///////////////////////////////////// Full restore ///////////////////////////////////////////////////////// sl@0: //////////////////////////////////////////////////////////////////////////////////////////////////////////////////// sl@0: sl@0: /** sl@0: Called when the BUE wants to start sending data to us. sl@0: Creates the folder (if the folder does not exist) where the temporary files will be created during the restore process. sl@0: Deletes all files from the restore folder. sl@0: sl@0: The restore is initiated by a notification received in CSqlBurEventMonitor::RunL() method. sl@0: InitialiseRestoreProxyBaseDataL() is called once per {client secure id, drive} and each sl@0: InitialiseRestoreProxyBaseDataLL() call is followed after that by a set of CSqlBurCallback::RestoreBaseDataSectionL() sl@0: calls, made from the backup and restore client dll. sl@0: sl@0: During RestoreBaseDataSectionLL() calls the CSqlBurCallback object will receive data from the backup and resore server. sl@0: sl@0: @see CSqlBurEventMonitor::RunL() sl@0: @see CSqlBurCallback::RestoreBaseDataSectionL() sl@0: sl@0: @param aSid the UID of the application that is to be restored. Not used (only logged in OST builds). sl@0: @param aDrive the drive to restore. sl@0: @leave KErrNoMemory, an out of memory condition has occurred; sl@0: Note that the function may also leave with some other system-wide error codes. sl@0: */ sl@0: void CSqlBurCallback::InitialiseRestoreProxyBaseDataL(TSecureId aSid, TDriveNumber aDrive) sl@0: { sl@0: UNUSED_ARG(aSid); sl@0: UNUSED_ARG(aDrive); sl@0: SQL_TRACE_BUR(OstTraceExt3(TRACE_INTERNALS, CSQLBACKUPCLIENT_INITIALIZERESTOREPROXYBASEDATAL, "0x%X;CSqlBurCallback::InitialiseRestoreProxyBaseDataL;aSid=0x%X;aDrive=%d", (TUint)this, (TUint)aSid.iId, (TInt)aDrive)); sl@0: iBuffer.Zero(); sl@0: iState = ERestoreExpectChecksum; sl@0: iRestoreDrive = aDrive; sl@0: iRestoreId = aSid; sl@0: //Create the directory for the temporary files created during the restore process. sl@0: TFileName privatePath; sl@0: __SQLLEAVE_IF_ERROR(iInterface.Fs().PrivatePath(privatePath)); sl@0: TDriveUnit driveUnit(iRestoreDrive); sl@0: TDriveName driveName = driveUnit.Name(); sl@0: privatePath.Insert(0, driveName); sl@0: __SQLLEAVE_IF_ERROR(iParse.Set(KSqlBurRestoreDir, &privatePath, 0)); sl@0: iRestoreDir.Copy(iParse.FullName()); sl@0: iRestoreDir.Append(KPathDelimiter); sl@0: TInt err = iInterface.Fs().MkDirAll(iRestoreDir); sl@0: if(err != KErrAlreadyExists) sl@0: { sl@0: __SQLLEAVE_IF_ERROR(err); sl@0: } sl@0: //Cleanup the restore directory sl@0: err = RestoreCleanup(); sl@0: if(err != KErrNotFound) sl@0: { sl@0: __SQLLEAVE_IF_ERROR(err); sl@0: } sl@0: } sl@0: sl@0: /** sl@0: This is called by BUE when the restore has completed. sl@0: sl@0: @see CSqlBurEventMonitor::RunL() sl@0: @see CSqlBurCallback::InitialiseRestoreProxyBaseDataL() sl@0: sl@0: @param aDrive the drive that is being restored. Not used (only logged in OST builds). sl@0: */ sl@0: void CSqlBurCallback::RestoreComplete(TDriveNumber aDrive) sl@0: { sl@0: UNUSED_ARG(aDrive); sl@0: SQL_TRACE_BUR(OstTraceExt2(TRACE_INTERNALS, CSQLBACKUPCLIENT_RESTORECOMPLETE, "0x%X;CSqlBurCallback::RestoreComplete;aDrive=%d", (TUint)this, (TInt)aDrive)); sl@0: iRestoreDrive = TDriveNumber(-1); sl@0: iRestoreId = TSecureId(KNullUid); sl@0: } sl@0: sl@0: /** sl@0: This is repeatedly called by the BUE to send us chunks of restore data (for the current SID) sl@0: Becuase the data is spread over chunks we need to manage the state across mutiple calls sl@0: to this method so we use a state machine. sl@0: sl@0: The function runs the state machine and for each file block detected in the coming data, the function does: sl@0: - creates a temporary file in the restore directory (created by InitialiseRestoreProxyBaseDataL()); sl@0: - stores the file data in the created temporary file; sl@0: - During the 2 steps descirbed above, if an error occurs, that erro will be reproted to the backup and restore sl@0: server (via a User::Leave() call); sl@0: - When all data is received and stored in temporary files in the restore directory, sl@0: for each received file the function will: sl@0: = move the original database file to the restore directory with a ".bak" extension added to the file name; sl@0: = move the temporary file, which has the same name as the original database file, to the location of the sl@0: original database file - the SQL server private data cage; sl@0: = delete the file with the ".bak" extension; sl@0: The three steps described above are implemented as "all or none" operation - if an error occurs during step (2), sl@0: the content of the original database file will be restored from the file with the ".bak" extension. sl@0: sl@0: @see CSqlBurEventMonitor::RunL() sl@0: @see CSqlBurCallback::InitialiseRestoreProxyBaseDataL() sl@0: sl@0: @param aInBuffer Buffer with data to be restored sl@0: @param aFinishedFlag Set when there is not more data to restore sl@0: @leave KErrNoMemory, an out of memory condition has occurred; sl@0: Note that the function may also leave with some other system-wide error codes. sl@0: */ sl@0: void CSqlBurCallback::RestoreBaseDataSectionL(TDesC8& aInBuffer, TBool aFinishedFlag) sl@0: { sl@0: // used to walk the buffer sl@0: // got a new buffer - because each time this method is called, we have a sl@0: // fresh chunk of data sl@0: TInt inBufferPos = 0; sl@0: sl@0: // to mark when the state machine is through sl@0: TBool done = EFalse; sl@0: sl@0: // check whether this is an empty restore sl@0: if(aFinishedFlag && aInBuffer.Size() == 0) sl@0: { sl@0: return; sl@0: } sl@0: sl@0: TInt iterations = 0; sl@0: sl@0: // run the state machine sl@0: do sl@0: { sl@0: // how many bytes are there available in the buffer for processing? sl@0: TInt bytesAvailable = aInBuffer.Size() - inBufferPos; sl@0: // the reason why we are testing finishedFlag is because we must sl@0: // make sure we re-enter the machine to do the tidyup sl@0: if(bytesAvailable <= 0 && !aFinishedFlag) sl@0: { sl@0: // ran out of data in the chunk sl@0: // so we return and wait for more data to arrive sl@0: return; sl@0: } sl@0: switch(iState) sl@0: { sl@0: case ERestoreExpectChecksum: // 16 bytes (the header is UTF16 encoded, 8 unicode characters for the checksum) sl@0: { sl@0: const TInt KCheckSumStrLen = 8; sl@0: CopyBufData(aInBuffer, inBufferPos, iBuffer, KCheckSumStrLen); sl@0: if(iBuffer.Length() == KCheckSumStrLen) sl@0: { sl@0: iChecksum = ::GetNumUint32L(iBuffer); sl@0: iState = ERestoreExpectOldFileSize; sl@0: iBuffer.Zero(); sl@0: } sl@0: break; sl@0: } sl@0: case ERestoreExpectOldFileSize: // 16 bytes (the header is UTF16 encoded, 8 unicode characters for 32-bit old file size) sl@0: { sl@0: const TInt KOldFileSizeStrLen = 8; sl@0: CopyBufData(aInBuffer, inBufferPos, iBuffer, KOldFileSizeStrLen); sl@0: if(iBuffer.Length() == KOldFileSizeStrLen) sl@0: { sl@0: TUint32 oldFileSize = ::GetNumUint32L(iBuffer); sl@0: if(oldFileSize == KSqlBurMagicNum) sl@0: { sl@0: iState = ERestoreExpectVersion; sl@0: } sl@0: else sl@0: { sl@0: iFileSize = oldFileSize; sl@0: iState = ERestoreExpectFileNameSize; sl@0: } sl@0: iBuffer.Zero(); sl@0: } sl@0: break; sl@0: } sl@0: case ERestoreExpectVersion: sl@0: { sl@0: const TInt KVersionStrLen = 4; sl@0: CopyBufData(aInBuffer, inBufferPos, iBuffer, KVersionStrLen); sl@0: if(iBuffer.Length() == KVersionStrLen) sl@0: { sl@0: //Ignore the version: ::GetNumUint32L(iBuffer); sl@0: //At this stage we know that the version is 2+ sl@0: iState = ERestoreExpectFileSize; sl@0: iBuffer.Zero(); sl@0: } sl@0: break; sl@0: } sl@0: case ERestoreExpectFileSize: sl@0: { sl@0: const TInt KFileSizeStrLen = 16; sl@0: CopyBufData(aInBuffer, inBufferPos, iBuffer, KFileSizeStrLen); sl@0: if(iBuffer.Length() == KFileSizeStrLen) sl@0: { sl@0: iFileSize = GetNumInt64L(iBuffer); sl@0: iState = ERestoreExpectFileNameSize; sl@0: iBuffer.Zero(); sl@0: } sl@0: SQL_TRACE_BUR(OstTraceExt2(TRACE_INTERNALS, CSQLBACKUPCLIENT_RESTOREBASEDATASECTONL1, "0x%X;CSqlBurCallback::RestoreBaseDataSectionL;iFileSize=%lld", (TUint)this, iFileSize)); sl@0: break; sl@0: } sl@0: case ERestoreExpectFileNameSize: // the size of the file name to restore sl@0: { sl@0: const TInt KFileNameLenStrLen = 8; sl@0: CopyBufData(aInBuffer, inBufferPos, iBuffer, KFileNameLenStrLen); sl@0: if(iBuffer.Length() == KFileNameLenStrLen) sl@0: { sl@0: iFileNameSize = GetNumUint32L(iBuffer); sl@0: iState = ERestoreExpectFileName; sl@0: iBuffer.Zero(); sl@0: } sl@0: break; sl@0: } sl@0: case ERestoreExpectFileName: // the name of the file to restore sl@0: { sl@0: CopyBufData(aInBuffer, inBufferPos, iBuffer, iFileNameSize); sl@0: SQL_TRACE_BUR(OstTraceExt4(TRACE_INTERNALS, CSQLBACKUPCLIENT_RESTOREBASEDATASECTONL2, "0x%X;CSqlBurCallback::RestoreBaseDataSectionL;BEGIN;iBuffer=%S;iBuffer.Length()=%d;iFileNameSize=%d", (TUint)this, __SQLPRNSTR(iBuffer), iBuffer.Length(), iFileNameSize)); sl@0: if(iBuffer.Length() == iFileNameSize) sl@0: { sl@0: iState = ERestoreExpectData; sl@0: TParse parse; sl@0: __SQLLEAVE_IF_ERROR(parse.Set(iBuffer, 0, 0)); sl@0: __SQLLEAVE_IF_ERROR(iParse.Set(parse.NameAndExt(), &iRestoreDir, 0)); sl@0: TPtrC fname(iParse.FullName()); sl@0: //The database is restored first to a temporary file, in the restore folder, on the same drive. sl@0: __SQLLEAVE_IF_ERROR(iFile.Replace(iInterface.Fs(), fname, EFileWrite | EFileShareExclusive)); sl@0: SQL_TRACE_BUR(OstTraceExt3(TRACE_INTERNALS, CSQLBACKUPCLIENT_RESTOREBASEDATASECTONL3, "0x%X;CSqlBurCallback::RestoreBaseDataSectionL;fname=%S;iFile.SubSessionHandle()=0x%X", (TUint)this, __SQLPRNSTR(fname), (TUint)iFile.SubSessionHandle())); sl@0: iBuffer.Zero(); sl@0: } sl@0: break; sl@0: } sl@0: case ERestoreExpectData: // now for the data sl@0: { sl@0: TInt len = Min((aInBuffer.Size() - inBufferPos), iFileSize); sl@0: TInt err = iFile.Write(aInBuffer.Mid(inBufferPos, len)); sl@0: if(err != KErrNone) sl@0: { sl@0: (void)RestoreCleanup(); sl@0: __SQLLEAVE(err); sl@0: } sl@0: inBufferPos += len; sl@0: iFileSize -= len; sl@0: if(iFileSize == 0) sl@0: { sl@0: iState = ERestoreComplete; sl@0: } sl@0: break; sl@0: } sl@0: case ERestoreComplete: // file completely restored sl@0: { sl@0: TUint64 checkSum64 = 0; sl@0: TInt restoreErr = iFile.Flush(); sl@0: if(restoreErr == KErrNone) sl@0: { sl@0: // calculate the checksum sl@0: restoreErr = CheckSum(iFile, checkSum64); sl@0: } sl@0: iFile.Close(); sl@0: if(restoreErr != KErrNone) sl@0: { sl@0: (void)RestoreCleanup(); sl@0: __SQLLEAVE(restoreErr); sl@0: } sl@0: SQL_TRACE_BUR(OstTraceExt2(TRACE_INTERNALS, CSQLBACKUPCLIENT_RESTOREBASEDATASECTONL4, "0x%X;CSqlBurCallback::RestoreBaseDataSectionL;END;iFile.SubSessionHandle()=0x%X", (TUint)this, (TUint)iFile.SubSessionHandle())); sl@0: TUint32 checkSum32 = checkSum64 & KMaxTUint32; sl@0: if(checkSum32 != iChecksum) sl@0: { sl@0: (void)RestoreCleanup(); sl@0: __SQLLEAVE(KErrCorrupt); sl@0: } sl@0: if((aInBuffer.Size() - inBufferPos) > 0) sl@0: {//There are bytes to be consumed in the input buffer sl@0: iState = ERestoreExpectChecksum; sl@0: break; sl@0: } sl@0: SQL_TRACE_BUR(OstTrace1(TRACE_INTERNALS, CSQLBACKUPCLIENT_RESTOREBASEDATASECTONL5, "0x%X;CSqlBurCallback::RestoreBaseDataSectionL;aFinishedFlag=ETrue", (TUint)this)); sl@0: //End of data. We have all data restored in the restore folder. sl@0: //The final step of the "restoring files" process consists of the following sub-steps: sl@0: // - Rename the database file to be restored to a file with ".bak" extension sl@0: // - Rename the file with the restored data to the database file sl@0: // - Delete the file with ".bak" extension sl@0: //Do not leave during the restore process! Restore as much files as possible. sl@0: //The only excpetion is TParse::Set() - if it fails it is a fatal error, the sl@0: //restored file path cannot be constructed. sl@0: __ASSERT_DEBUG(iRestoreDrive != TDriveNumber(-1), __SQLPANIC(ESqlPanicInternalError)); sl@0: __ASSERT_DEBUG(iRestoreId != TSecureId(KNullUid), __SQLPANIC(ESqlPanicInternalError)); sl@0: //Include the aUid and the "*" mask sl@0: TUidName uidName = (static_cast (iRestoreId)).Name(); sl@0: TBuf fileNameMask(uidName); sl@0: fileNameMask.Append(KSqlBurAllFiles); sl@0: __SQLLEAVE_IF_ERROR(iParse.Set(fileNameMask, &iRestoreDir, 0)); sl@0: CDir* dir = NULL; sl@0: TPtrC searchPattern(iParse.FullName()); sl@0: SQL_TRACE_BUR(OstTraceExt2(TRACE_INTERNALS, CSQLBACKUPCLIENT_RESTOREBASEDATASECTONL55, "0x%X;CSqlBurCallback::RestoreBaseDataSectionL;search pattern=%S", (TUint)this, __SQLPRNSTR(searchPattern))); sl@0: restoreErr = iInterface.Fs().GetDir(searchPattern, KEntryAttNormal, ESortNone, dir); sl@0: if(restoreErr == KErrNone) sl@0: { sl@0: SQL_TRACE_BUR(OstTraceExt2(TRACE_INTERNALS, CSQLBACKUPCLIENT_RESTOREBASEDATASECTONL6, "0x%X;CSqlBurCallback::RestoreBaseDataSectionL;restored files=%d", (TUint)this, dir->Count())); sl@0: for(TInt i=0;iCount();++i) sl@0: { sl@0: const TEntry& entry = (*dir)[i]; sl@0: __SQLLEAVE_IF_ERROR(iParse.Set(entry.iName, &iRestoreDir, 0)); sl@0: TFileName dbName(iParse.FullName()); sl@0: SQL_TRACE_BUR(OstTraceExt2(TRACE_INTERNALS, CSQLBACKUPCLIENT_RESTOREBASEDATASECTONL7, "0x%X;CSqlBurCallback::RestoreBaseDataSectionL;restored file=%S", (TUint)this, __SQLPRNSTR(dbName))); sl@0: TInt pos = dbName.Find(KSqlBurRestoreDir); sl@0: __ASSERT_DEBUG(pos >= 0, __SQLPANIC(ESqlPanicInternalError)); sl@0: dbName.Delete(pos, KSqlBurRestoreDir().Length() + 1);//"+1" for the path delimitier sl@0: SQL_TRACE_BUR(OstTraceExt2(TRACE_INTERNALS, CSQLBACKUPCLIENT_RESTOREBASEDATASECTONL8, "0x%X;CSqlBurCallback::RestoreBaseDataSectionL;database=%S", (TUint)this, __SQLPRNSTR(dbName))); sl@0: TFileName bakDbName(iParse.FullName()); sl@0: bakDbName.Append(KSqlBurBackupExt); sl@0: SQL_TRACE_BUR(OstTraceExt2(TRACE_INTERNALS, CSQLBACKUPCLIENT_RESTOREBASEDATASECTONL9, "0x%X;CSqlBurCallback::RestoreBaseDataSectionL;backup file=%S", (TUint)this, __SQLPRNSTR(dbName))); sl@0: //Now, dbName contains the original database (full path), iParse - the restored file, sl@0: //bakDbName - backup file name sl@0: TInt err = iInterface.Fs().Rename(dbName, bakDbName); sl@0: if(err == KErrNone || err == KErrNotFound) sl@0: { sl@0: err = iInterface.Fs().Rename(iParse.FullName(), dbName); sl@0: if(err == KErrNone) sl@0: {//commit: delete the backup database file sl@0: SQL_TRACE_BUR(OstTraceExt2(TRACE_INTERNALS, CSQLBACKUPCLIENT_RESTOREBASEDATASECTONL10, "0x%X;CSqlBurCallback::RestoreBaseDataSectionL;Commit;file=%S", (TUint)this, __SQLPRNSTR(dbName))); sl@0: (void)iInterface.Fs().Delete(bakDbName); sl@0: } sl@0: else sl@0: {//rollback: restore the original database file sl@0: err = iInterface.Fs().Rename(bakDbName, dbName); sl@0: SQL_TRACE_BUR(OstTraceExt3(TRACE_INTERNALS, CSQLBACKUPCLIENT_RESTOREBASEDATASECTONL11, "0x%X;CSqlBurCallback::RestoreBaseDataSectionL;Rollback;file=%S;err=%d", (TUint)this, __SQLPRNSTR(dbName), err)); sl@0: } sl@0: } sl@0: if(err != KErrNone && err != KErrNotFound) sl@0: { sl@0: if(restoreErr == KErrNone) sl@0: { sl@0: restoreErr = err; sl@0: } sl@0: } sl@0: }//for(...) sl@0: delete dir; sl@0: }//iInterface.Fs().GetDir(...) sl@0: done = ETrue; sl@0: (void)RestoreCleanup(); sl@0: if(restoreErr != KErrNone) sl@0: { sl@0: __SQLLEAVE(restoreErr); sl@0: } sl@0: break; sl@0: } sl@0: default: sl@0: __ASSERT_DEBUG(EFalse, __SQLPANIC(ESqlPanicInternalError)); sl@0: break; sl@0: }//switch(...) sl@0: if((aInBuffer.Size() - inBufferPos) == bytesAvailable) sl@0: {//No bytes have been consumed from the buffer. sl@0: if(++iterations > 1 && !done) sl@0: {//This is the second iteration in the loop where no bytes have been consumed from the input buffer. sl@0: //But the "done" flag is still false. Corrupted archive. sl@0: __SQLLEAVE(KErrCorrupt); sl@0: } sl@0: } sl@0: } while(!done); sl@0: } sl@0: sl@0: /** sl@0: The operation was terminated - we should tidyup here (as best we can) sl@0: Backup: close the file, free the allocated memory for the file names. sl@0: Restore: since the final restore step is a non-leaving one, nothing special needs to be done here - sl@0: RestoreCleanup() is called to close the file and delete if there are any temporary files left. sl@0: */ sl@0: void CSqlBurCallback::TerminateMultiStageOperation() sl@0: { sl@0: BackupCleanup(); sl@0: (void)RestoreCleanup(); sl@0: } sl@0: sl@0: /** sl@0: We do our own checksumming so we don't need this sl@0: @return the checksum sl@0: @param aDrive the drive affected (unused) sl@0: */ sl@0: TUint CSqlBurCallback::GetDataChecksum(TDriveNumber /* aDrive */) sl@0: { sl@0: // not required - not implemented sl@0: const TUint KArbitraryNumber = 1024; sl@0: return KArbitraryNumber; sl@0: } sl@0: sl@0: //////////////////////////////////////////////////////////////////////////////////////////////////////////////////// sl@0: /////////////////////////// Incremental backup/restore //////////////////////////////////////////////////// sl@0: //////////////////////////////////////////////////////////////////////////////////////////////////////////////////// sl@0: sl@0: /** sl@0: We don't support incremental backup sl@0: */ sl@0: void CSqlBurCallback::GetSnapshotDataL(TDriveNumber /* aDrive */, TPtr8& /* aBuffer */, TBool& /* aFinishedFlag */) sl@0: { sl@0: __SQLLEAVE(KErrNotSupported); sl@0: } sl@0: sl@0: /** sl@0: We don't support incremental backup sl@0: */ sl@0: void CSqlBurCallback::InitialiseGetBackupDataL(TDriveNumber /* aDrive */) sl@0: { sl@0: __SQLLEAVE(KErrNotSupported); sl@0: } sl@0: sl@0: /** sl@0: We don't support incremental backup sl@0: */ sl@0: void CSqlBurCallback::InitialiseRestoreBaseDataL(TDriveNumber /* aDrive */) sl@0: { sl@0: __SQLLEAVE(KErrNotSupported); sl@0: } sl@0: sl@0: /** sl@0: We don't support incremental backup sl@0: */ sl@0: void CSqlBurCallback::InitialiseRestoreIncrementDataL(TDriveNumber /* aDrive */) sl@0: { sl@0: __SQLLEAVE(KErrNotSupported); sl@0: } sl@0: sl@0: /** sl@0: We don't support incremental backup sl@0: */ sl@0: void CSqlBurCallback::RestoreIncrementDataSectionL(TDesC8& /* aBuffer */, TBool /* aFinishedFlag */) sl@0: { sl@0: __SQLLEAVE(KErrNotSupported); sl@0: } sl@0: sl@0: /** sl@0: We don't support incremental backup sl@0: */ sl@0: void CSqlBurCallback::AllSnapshotsSuppliedL() sl@0: { sl@0: } sl@0: sl@0: /** sl@0: We don't support incremental backup sl@0: */ sl@0: void CSqlBurCallback::ReceiveSnapshotDataL(TDriveNumber /* aDrive */, TDesC8& /* aBuffer */, TBool /* aFinishedFlag */) sl@0: { sl@0: __SQLLEAVE(KErrNotSupported); sl@0: } sl@0: sl@0: //////////////////////////////////////////////////////////////////////////////////////////////////////////////////// sl@0: /////////////////////////// Helper functions ////////////////////////////////////////////////////////////// sl@0: //////////////////////////////////////////////////////////////////////////////////////////////////////////////////// sl@0: sl@0: /** sl@0: A simple checksumming algorithm to allow a degree sl@0: of trust that the backup and restore worked. sl@0: Note the file pointer will be back at the start when the function call completes successfully. sl@0: In case of an error, the position of the file pointer is undetermined. sl@0: sl@0: @param aOpenFile Already opened database file on which the checksum is calculated. sl@0: @param aCheckSum Output parameter. The checksum is returned in this parameter. sl@0: @return KErrNoMemory, an out of memory condition has occurred; sl@0: Note that the function may also return some other system-wide error codes. sl@0: */ sl@0: TInt CSqlBurCallback::CheckSum(const RFile64& aOpenFile, TUint64& aCheckSum) const sl@0: { sl@0: // scoot through the database file building the checksum sl@0: aCheckSum = 0; sl@0: TInt64 seekPos = 0; // rewind first sl@0: TInt err = aOpenFile.Seek(ESeekStart, seekPos); sl@0: if(err != KErrNone) sl@0: { sl@0: return err; sl@0: } sl@0: const TUint KCheckSumBlockSize = 4 * 1024; sl@0: HBufC8* buf = HBufC8::New(KCheckSumBlockSize); sl@0: if(!buf) sl@0: { sl@0: return KErrNoMemory; sl@0: } sl@0: TPtr8 ptr = buf->Des(); sl@0: for(;;) sl@0: { sl@0: err = aOpenFile.Read(ptr); sl@0: if(err != KErrNone) sl@0: { sl@0: delete buf; sl@0: return err; sl@0: } sl@0: TInt len = ptr.Length(); sl@0: if(len == 0) sl@0: { sl@0: break; sl@0: } sl@0: // calculate the checksum sl@0: for(TInt i=0;i> 63); sl@0: aCheckSum += ptr[i]; sl@0: } sl@0: }; sl@0: delete buf; sl@0: // restore file position sl@0: seekPos = 0; sl@0: err = aOpenFile.Seek(ESeekStart,seekPos); sl@0: return err; sl@0: } sl@0: sl@0: /** sl@0: Reads the content of aInBuf from position aInBufReadPos and stores the data into aOutBuf. sl@0: aDataLen is the length of the data. If the input buffer does not contain all the data, then only the sl@0: available data will be copied to the output buffer. sl@0: sl@0: How the function works. It is called during the restore process and aInBuf parameter contains a block of raw sl@0: data sent by the B&R server. The calling function, RestoreBaseDataSectionL(), uses a state sl@0: machine to processes the incoming data. At particular moment RestoreBaseDataSectionL() will process the data header sl@0: and will have to read "aDataLen" 16-bit characters at position "aInBufReadPos". If there are "aDataLen" characters sl@0: at position "aInBufReadPos" and enough free space in "aOutBuf", CopyBufData() will copy all of them, sl@0: otherwise CopyBufData() will copy as much characters as possible (in which case RestoreBaseDataSectionL() will sl@0: stay in the same state, waiting for more data from the B&R server). sl@0: sl@0: @param aInBuf 8-bit buffer with input data sl@0: @param aInBufReadPos The position in the buffer from which the read operation starts. sl@0: When the "buffer read" operatio completes, aInBufReadPos is updated with the sl@0: number of bytes read from the input buffer. sl@0: @param aOutBuf 16-bit output buffer. The data read from the input buffer is stored in the output buffer. sl@0: @param aDataLen How much bytes to be read from the input buffer. Note that if there is not enough sl@0: data in the input buffer, the function will read as much as possible from the input buffer. sl@0: The aInBufReadPos in/out parameter will be updated with the actual number of bytes read. sl@0: */ sl@0: void CSqlBurCallback::CopyBufData(const TDesC8& aInBuf, TInt& aInBufReadPos, TDes& aOutBuf, TInt aDataLen) sl@0: { sl@0: __ASSERT_DEBUG(aInBufReadPos >= 0, __SQLPANIC(ESqlPanicBadArgument)); sl@0: __ASSERT_DEBUG(aDataLen > 0, __SQLPANIC(ESqlPanicBadArgument)); sl@0: sl@0: TInt needed = (aDataLen - aOutBuf.Length()) << K8to16bitShift; sl@0: TInt available = aInBuf.Size() - aInBufReadPos; sl@0: TInt len = Min(needed, available); sl@0: TPtrC8 ptr8 = aInBuf.Mid(aInBufReadPos, len); sl@0: aInBufReadPos += len; sl@0: sl@0: len >>= K8to16bitShift; sl@0: aOutBuf.Append((const TUint16*)ptr8.Ptr(), len); sl@0: } sl@0: sl@0: /** sl@0: Cleans up the allocated during the backup resources - file handles, buffers allocated for the file names. sl@0: */ sl@0: void CSqlBurCallback::BackupCleanup() sl@0: { sl@0: for(TInt i=0;iDelete(allFiles); sl@0: delete fm; sl@0: } sl@0: return err; sl@0: } sl@0: sl@0: /** sl@0: Stores the error occured during backup for furhter processing. sl@0: Please note that the function asserts if the aError parameter is KErrNone. sl@0: Call the function only with a real error. sl@0: sl@0: @param aError The backup error to be stored sl@0: */ sl@0: void CSqlBurCallback::SetBackupError(TInt aError) sl@0: { sl@0: __ASSERT_DEBUG(aError != KErrNone, __SQLPANIC(ESqlPanicBadArgument)); sl@0: if(aError != KErrNotFound && aError != KErrPathNotFound) sl@0: { sl@0: if(iBackupError == KErrNone || aError == KErrDiskFull || aError == KErrCorrupt) sl@0: { sl@0: iBackupError = aError; sl@0: } sl@0: } sl@0: }