backupandrestore/backupengine/src/sbesession.cpp
author Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
Tue, 25 May 2010 13:35:25 +0300
branchRCL_3
changeset 24 523717cdb0ad
parent 0 d0791faffa3f
child 29 3ae5cb0b4c02
permissions -rw-r--r--
Revision: 201004 Kit: 2010121

// Copyright (c) 2004-2009 Nokia Corporation and/or its subsidiary(-ies).
// All rights reserved.
// This component and the accompanying materials are made available
// under the terms of "Eclipse Public License v1.0"
// which accompanies this distribution, and is available
// at the URL "http://www.eclipse.org/legal/epl-v10.html".
//
// Initial Contributors:
// Nokia Corporation - initial contribution.
//
// Contributors:
//
// Description:
// Implementation of CSBESession class.
// 
//

/**
 @file
*/

#include <e32std.h>
#include <e32base.h>
#include "sbeserver.h"
#include "sbesession.h"
#include "sbeclientserver.h"
#include <connect/panic.h>
#include "sbedataownermanager.h"
#include <connect/sbtypes.h>
#include "sblog.h"
#include <apgcli.h>

namespace conn
	{
	
	/** This block size for the internal buffer
	@internalTechnology
	*/
	const TInt KSBBufferBlockSize = 4096;
	
	CSBESession::CSBESession() : iTransferBuf(NULL), iTransferTextBuf(NULL), iArrayCount(0)
    /**
    Class Constructor
    */
		{
		}

	CSBESession::~CSBESession()
    /**
    Class destructor
    */
		{
		//
		// If the client has detached properly, they should
		// have done this - but just in case.
		Server().DropSession();
		ResetTransferBuf();
		delete iExtCurEntry;
		}

	void CSBESession::CreateL()
	/**
	Creates a connection between the Secure Backup Engine server and the Secure
	Backup Engine session.  Increments the server's session count
	*/
		{
		//
		// Increase the servers session count.
		Server().AddSession();
		}

	void CSBESession::ServiceL(const RMessage2& aMessage)
	/**
	Called by the client server framework to service a message request
	from a client.

    @param aMessage  Reference to a RMessage2 object
	*/
		{
	#ifdef SBE_LOGGING_ENABLED
		RThread thread;
		aMessage.Client(thread);
		const TFullName threadName( thread.FullName() );
		thread.Close();
	#endif		
		
		switch(aMessage.Function())
			{
			//
			// Connection config getting/setting.
			case ESBEMsgGetGSHHandle:
				{
        		__LOG1("CSBESession::ServiceL() - ESBEMsgGetGSHHandle for thread: %S", &threadName);
				aMessage.Complete(Server().GlobalSharedHeap());
				break;
				}
			case ESBEMsgPrepDataOwnerInfo:
				{
        		__LOG1("CSBESession::ServiceL() - ESBEMsgPrepDataOwnerInfo for thread: %S", &threadName);
				PrepDataOwnerInfoL(aMessage);
				break;
				}
			case ESBEMsgGetDataOwnerInfo:
				{
        		__LOG1("CSBESession::ServiceL() - ESBEMsgGetDataOwnerInfo for thread: %S", &threadName);
				ReturnDataOwnerInfoL(aMessage);
				break;
				}
			case ESBEMsgPrepPublicFiles:
				{
        		__LOG1("CSBESession::ServiceL() - ESBEMsgPrepPublicFiles for thread: %S", &threadName);
				PrepPublicFileListL(aMessage);
				break;
				}
			case ESBEMsgGetPublicFiles:
				{
        		__LOG1("CSBESession::ServiceL() - ESBEMsgGetPublicFiles for thread: %S", &threadName);
				ReturnPublicFileListL(aMessage);
				break;
				}
			case ESBEMsgPrepPublicFilesRaw:
				{
        		__LOG1("CSBESession::ServiceL() - ESBEMsgPrepPublicFilesRaw for thread: %S", &threadName);
				PrepPublicFileListRawL(aMessage);
				break;
				}
			case ESBEMsgGetPublicFilesRaw:
				{
        		__LOG1("CSBESession::ServiceL() - ESBEMsgGetPublicFilesRaw for thread: %S", &threadName);
				ReturnPublicFileListRawL(aMessage);
				break;
				}
			case ESBEMsgPrepPublicFilesXML:
				{
        		__LOG1("CSBESession::ServiceL() - ESBEMsgPrepPublicFilesXML for thread: %S", &threadName);
				PrepPublicFileListXMLL(aMessage);
				break;
				}
			case ESBEMsgGetPublicFilesXML:
				{
        		__LOG1("CSBESession::ServiceL() - ESBEMsgGetPublicFilesXML for thread: %S", &threadName);
				ReturnPublicFileListXMLL(aMessage);
				break;
				}
			case ESBEMsgSetBURMode:
				{
        		__LOG1("CSBESession::ServiceL() - ESBEMsgSetBURMode for thread: %S", &threadName);
				SetBURModeL(aMessage);
				break;
				}
			case ESBEMsgSetSIDListPartial:
				{
        		__LOG1("CSBESession::ServiceL() - ESBEMsgSetSIDListPartial for thread: %S", &threadName);
				SetSIDListForPartialBURL(aMessage);
				break;
				}
			case ESBEMsgPrepSIDStatus:
				{
        		__LOG1("CSBESession::ServiceL() - ESBEMsgPrepSIDStatus for thread: %S", &threadName);
				PrepSIDStatusL(aMessage);
				break;
				}
			case ESBEMsgGetSIDStatus:
				{
        		__LOG1("CSBESession::ServiceL() - ESBEMsgGetSIDStatus for thread: %S", &threadName);
				ReturnSIDStatusL(aMessage);
				break;
				}
			case ESBEMsgRequestDataSync:
			case ESBEMsgRequestDataAsync:
				{
        		__LOG1("CSBESession::ServiceL() - ESBEMsgRequestDataSync / ESBEMsgRequestDataAsync for thread: %S", &threadName);
				TRAPD(reqDataErr, RequestDataAsyncL(aMessage));
				if (reqDataErr != KErrNone)
					{
					Server().GSHInterface().Header(Server().GlobalSharedHeap()).SetLockedFlag(EFalse);
					User::Leave(reqDataErr);
					}
				break;
				}
			case ESBEMsgSupplyDataSync:
				{
        		__LOG1("CSBESession::ServiceL() - ESBEMsgSupplyDataSync for thread: %S", &threadName);
				TRAPD(supDataErr, SupplyDataSyncL(aMessage));
				if (supDataErr != KErrNone)
					{
					Server().GSHInterface().Header(Server().GlobalSharedHeap()).SetLockedFlag(EFalse);
					User::Leave(supDataErr);
					}
				break;
				}
			case ESBEMsgAllSnapshotsSupplied:
				{
        		__LOG1("CSBESession::ServiceL() - ESBEMsgAllSnapshotsSupplied for thread: %S", &threadName);
				AllSnapshotsSuppliedL(aMessage);
				break;
				}
			case ESBEMsgGetExpectedDataSize:
				{
        		__LOG1("CSBESession::ServiceL() - ESBEMsgGetExpectedDataSize for thread: %S", &threadName);
				GetExpectedDataSizeL(aMessage);
				break;
				}
			case ESBEMsgAllSystemFilesRestored:
				{
        		__LOG1("CSBESession::ServiceL() - ESBEMsgAllSystemFilesRestored for thread: %S", &threadName);
				AllSystemFilesRestoredL();
				aMessage.Complete(KErrNone);
				break;
				}
			case ESBEMsgPrepLargePublicFiles:
				{
        		__LOG1("CSBESession::ServiceL() - ESBEMsgPrepPublicFiles for thread: %S", &threadName);
				PrepLargePublicFileListL(aMessage);
				break;
				}
			case ESBEMsgGetLargePublicFiles:
				{
        		__LOG1("CSBESession::ServiceL() - ESBEMsgGetPublicFiles for thread: %S", &threadName);
				ReturnLargePublicFileListL(aMessage);
				break;
				}
			default:
				{
        		__LOG1("CSBESession::ServiceL() - UNKNOWN OP CODE for thread: %S", &threadName);
				User::Leave(KErrNotSupported);
				}
			}

		__LOG1("CSBESession::ServiceL() - Completed OK for thread: %S", &threadName);
		}

	inline CSBEServer& CSBESession::Server() const
	/**
	Returns a non-cost reference to this CServer object.

	@return The non-const reference to this.
	*/
		{
		return *static_cast<CSBEServer*>(const_cast<CServer2*>(CSession2::Server()));
		}
		
	void CSBESession::PrepDataOwnerInfoL(const RMessage2& aMessage)
	/**
	Prepares the 
	*/
		{
		TInt err = KErrNone;
		ResetTransferBuf();
		RPointerArray<CDataOwnerInfo> doiArray;
		TRAP(err,
			{
			Server().DataOwnerManager().GetDataOwnersL(doiArray);

			iTransferBuf = HBufC8::NewL(KSBBufferBlockSize);
			TPtr8 transferBufPtr(iTransferBuf->Des());
			
			iArrayCount = doiArray.Count();
			
			HBufC8* dataOwnerBuffer = NULL;
			for (TInt index = 0; index < iArrayCount; index++)
				{
				dataOwnerBuffer = doiArray[index]->ExternaliseLC();
				
				if (transferBufPtr.Size() + dataOwnerBuffer->Size() > transferBufPtr.MaxSize())
					{
					iTransferBuf = iTransferBuf->ReAllocL(transferBufPtr.MaxSize() + KSBBufferBlockSize);
					transferBufPtr.Set(iTransferBuf->Des());
					}
					
				// Append the flattened data owner to the IPC transfer buffer	
				transferBufPtr.Append(*dataOwnerBuffer);
				
				CleanupStack::PopAndDestroy(dataOwnerBuffer);
				}
			}
			); // TRAP
		
		doiArray.ResetAndDestroy();
		doiArray.Close();

		User::LeaveIfError(err);
		
		// complete the message with the total size of the buffer so that the 
		// client can create the appropriate sized descriptor for receiving the data
		aMessage.Complete(iTransferBuf->Des().Size());
		}
		
	void CSBESession::ReturnDataOwnerInfoL(const RMessage2& aMessage)
	/**
	Return the previously populated buffer to the client
	@param aMessage The message sent by the client to the server
	*/
		{
		__ASSERT_DEBUG(iTransferBuf, Panic(KErrNotFound));
		
		// return the previously allocated transfer buffer
		aMessage.WriteL(0, *iTransferBuf);		
		aMessage.Complete(iArrayCount);
		
		ResetTransferBuf();
		}
		
	void CSBESession::PrepPublicFileListL(const RMessage2& aMessage)
	/**
	Compile a list of public files owned by a particular SID to be backed up
	@param aMessage The message sent by the client to the server
	*/
		{
		RFileArray fileArray;
		CleanupClosePushL(fileArray);
		HBufC8* pGenericDataTypeBuffer = HBufC8::NewLC(aMessage.GetDesLengthL(1));
		
		TPtr8 genericDataTypeBuffer(pGenericDataTypeBuffer->Des());
		aMessage.ReadL(1, genericDataTypeBuffer);
		
		TDriveNumber drive = static_cast<TDriveNumber>(aMessage.Int0());
		
		CSBGenericDataType* pGenericDataType = CSBGenericDataType::NewL(*pGenericDataTypeBuffer);
		CleanupStack::PopAndDestroy(pGenericDataTypeBuffer);
		CleanupStack::PushL(pGenericDataType);
		Server().DataOwnerManager().GetPublicFileListL(pGenericDataType, drive, fileArray);
		CleanupStack::PopAndDestroy(pGenericDataType);
		
		iTransferBuf = fileArray.ExternaliseL();

		CleanupStack::PopAndDestroy(&fileArray);

		// complete the message with the total size of the buffer so that the 
		// client can create the appropriate sized descriptor for receiving the data
		aMessage.Complete(iTransferBuf->Des().MaxSize());
		}
		
	void CSBESession::ReturnPublicFileListL(const RMessage2& aMessage)
	/** Return the previously populated buffer to the client
	@param aMessage The message sent by the client to the server */
		{
		__ASSERT_DEBUG(iTransferBuf, Panic(KErrArgument));
		
		// return the previously allocated transfer buffer
		aMessage.WriteL(0, *iTransferBuf);
		
		aMessage.Complete(KErrNone);
		
		ResetTransferBuf();
		}

	void CSBESession::PrepPublicFileListRawL(const RMessage2& aMessage)
	/**
	Compile a raw list of public files owned by a particular SID to be backed up
	@param aMessage The message sent by the client to the server
	*/
		{
		RRestoreFileFilterArray fileFilterArray;
		CleanupClosePushL(fileFilterArray);
		TPckgBuf<TDriveNumber> driveNumPkg;
		HBufC8* pGenericDataTypeBuffer = HBufC8::NewLC(aMessage.GetDesLengthL(1));
		
		TPtr8 genericDataTypeBuffer(pGenericDataTypeBuffer->Des());
		
		aMessage.ReadL(0, driveNumPkg);
		aMessage.ReadL(1, genericDataTypeBuffer);
		
		CSBGenericDataType* pGenericDataType = CSBGenericDataType::NewL(*pGenericDataTypeBuffer);
		CleanupStack::PopAndDestroy(pGenericDataTypeBuffer);
		CleanupStack::PushL(pGenericDataType);
		
		Server().DataOwnerManager().GetRawPublicFileListL(pGenericDataType, driveNumPkg(), fileFilterArray);
		CleanupStack::PopAndDestroy(pGenericDataType);
		
		iTransferBuf = fileFilterArray.ExternaliseL();

		CleanupStack::PopAndDestroy(&fileFilterArray);

		// complete the message with the total size of the buffer so that the 
		// client can create the appropriate sized descriptor for receiving the data
		aMessage.Complete(iTransferBuf->Des().MaxSize());
		}
		
	void CSBESession::ReturnPublicFileListRawL(const RMessage2& aMessage)
	/** Return the previously populated buffer to the client
	@param aMessage The message sent by the client to the server */
		{
		__ASSERT_DEBUG(iTransferBuf, Panic(KErrArgument));
		
		// return the previously allocated transfer buffer
		aMessage.WriteL(0, *iTransferBuf);
		
		aMessage.Complete(KErrNone);
		
		ResetTransferBuf();
		}

	void CSBESession::PrepPublicFileListXMLL(const RMessage2& aMessage)
	/**
	*/
		{
		TPckgBuf<TDriveNumber> driveNumPkg;
		TPckgBuf<TSecureId> sidPkg;

		ResetTransferBuf();

		iTransferTextBuf = HBufC::NewL(0);

		Server().DataOwnerManager().GetXMLPublicFileListL(sidPkg(), driveNumPkg(), iTransferTextBuf);
		
		iTransferBuf = HBufC8::NewL(0);
		
		aMessage.Complete(iTransferBuf->Des().MaxSize());
		}
		
	void CSBESession::ReturnPublicFileListXMLL(const RMessage2& aMessage)
	/**
	Return the previously populated buffer to the client
	@param aMessage The message sent by the client to the server
	*/
		{
		__ASSERT_DEBUG(iTransferBuf, Panic(KErrArgument));
		
		// return the previously allocated transfer buffer
		aMessage.WriteL(0, *iTransferTextBuf);

		aMessage.Complete(KErrNone);
		
		ResetTransferBuf();
		}
		
	void CSBESession::SetBURModeL(const RMessage2& aMessage)
	/**	
	Set the Backup and Restore mode on/off and configure the BUR options

	@param aMessage The message sent by the client to the server
	*/
		{
		TDriveList driveList;
		
		aMessage.ReadL(0, driveList);
		TBURPartType burType = static_cast<TBURPartType>(aMessage.Int1());
		TBackupIncType incType = static_cast<TBackupIncType>(aMessage.Int2());
		
		
		// Unlock the global heap
		Server().GSHInterface().ResetHeap(Server().GlobalSharedHeap());

		Server().DataOwnerManager().SetBURModeL(driveList, burType, incType);
		
		aMessage.Complete(KErrNone);
		}

	void CSBESession::SetSIDListForPartialBURL(const RMessage2& aMessage)
	/**
	Set a list of SID's that require base backup regardless of the device backup mode
	
	@param aMessage The message sent by the client to the server
	*/
		{
		HBufC8* flatArray = HBufC8::NewLC(aMessage.GetDesLengthL(0));	
		TPtr8 flatArrayPtr(flatArray->Des());
		
		aMessage.ReadL(0, flatArrayPtr);

		Server().DataOwnerManager().SetSIDListForPartialBURL(flatArrayPtr);

		CleanupStack::PopAndDestroy(flatArray);
		
		aMessage.Complete(KErrNone);
		}

	void CSBESession::PrepSIDStatusL(const RMessage2& aMessage)
	/**
	Extract the array of SID's of which the status is required
	
	@param aMessage The message sent from the client to the server
	*/
		{
		ResetTransferBuf();
		iTransferBuf = HBufC8::NewL(aMessage.GetDesLengthL(0));
		
		TPtr8 transBuf(iTransferBuf->Des());
		// Copy the IPC'd buffer into our transfer buffer
		aMessage.ReadL(0, transBuf);
		
		aMessage.Complete(KErrNone);
		}
		
	void CSBESession::ReturnSIDStatusL(const RMessage2& aMessage)
	/**
	Populate and return the array of SID's complete with their statuses
	
	@param aMessage The message sent from the client to the server
	*/
		{
		RSIDStatusArray* pStatusArray = RSIDStatusArray::InternaliseL(*iTransferBuf);
		CleanupStack::PushL(pStatusArray);
		CleanupClosePushL(*pStatusArray);
		
		// We're finished with the received externalised SIDStatusArray for now, delete it
		ResetTransferBuf();
		
		// Ask DOM to populate the Statuses in the status array
		Server().DataOwnerManager().SIDStatusL(*pStatusArray);
		
		iTransferBuf = pStatusArray->ExternaliseL();
		
		CleanupStack::PopAndDestroy(2, pStatusArray);
		
		aMessage.WriteL(0, *iTransferBuf);		
		aMessage.Complete(KErrNone);

		ResetTransferBuf();
		}
		
	void CSBESession::RequestDataAsyncL(const RMessage2& aMessage)
	/**
	Handle the client's asynchronous request for data from the SBE
	*/
		{
		iMessage = aMessage;
		
        __LOG("CSBESession::RequestDataAsyncL() - START");
		TBool finished = EFalse;	// Set by DOM, initialised to eliminate warning
		HBufC8* pTransferredBuf = HBufC8::NewLC(iMessage.GetDesLengthL(0));
		
		TPtr8 transBuf(pTransferredBuf->Des());
		// Copy the IPC'd buffer into our transfer buffer
		iMessage.ReadL(0, transBuf);
		
		// Extract the generic type from the client IPC request 
		CSBGenericTransferType* pGenericType = CSBGenericTransferType::NewL(*pTransferredBuf);
		CleanupStack::PopAndDestroy(pTransferredBuf);
		CleanupStack::PushL(pGenericType);
		
		TPtr8& writeBuf = Server().GSHInterface().WriteBufferL(Server().GlobalSharedHeap());

		Server().GSHInterface().Header(Server().GlobalSharedHeap()).SetLockedFlag(ETrue);
		
		Server().GSHInterface().Header(Server().GlobalSharedHeap()).GenericTransferTypeBuffer() = pGenericType->Externalise();
		
		// Call the DOM to populate the GSH with the data to return
		Server().DataOwnerManager().RequestDataL(pGenericType, writeBuf, finished);
		
		CleanupStack::PopAndDestroy(pGenericType);
		
		Server().GSHInterface().Header(Server().GlobalSharedHeap()).iFinished = finished;
        __LOG2("CSBESession::RequestDataAsyncL() - Server-side data length: %d, address: 0x%08x", writeBuf.Length(), writeBuf.Ptr());
        //__LOGDATA("CSBESession::RequestDataAsyncL() -       %S", writeBuf.Ptr(), writeBuf.Length());

		Server().GSHInterface().Header(Server().GlobalSharedHeap()).SetLockedFlag(EFalse);
		
		if (!iMessage.IsNull())
			{
			iMessage.Complete(KErrNone);
			}
		
        __LOG("CSBESession::RequestDataAsyncL() - END");
		}
		
	void CSBESession::RequestDataSyncL(const RMessage2& aMessage)
	/**
	Handle the client's synchronous request for data from the SBE
	*/
		{
        __LOG("CSBESession::RequestDataSyncL() - START");
		TBool finished = EFalse;	// Set by DOM, initialised to eliminate warning
		HBufC8* pTransferredBuf = HBufC8::NewLC(aMessage.GetDesLengthL(0));
		
		TPtr8 transBuf(pTransferredBuf->Des());
		// Copy the IPC'd buffer into our transfer buffer
		aMessage.ReadL(0, transBuf);
		
		// Extract the generic type from the client IPC request 
		CSBGenericTransferType* pGenericType = CSBGenericTransferType::NewL(*pTransferredBuf);
		CleanupStack::PopAndDestroy(pTransferredBuf);
		CleanupStack::PushL(pGenericType);
		
		TPtr8& writeBuf = Server().GSHInterface().WriteBufferL(Server().GlobalSharedHeap());

		Server().GSHInterface().Header(Server().GlobalSharedHeap()).SetLockedFlag(ETrue);

		Server().GSHInterface().Header(Server().GlobalSharedHeap()).GenericTransferTypeBuffer() = pGenericType->Externalise();
		
		// Call the DOM to populate the GSH with the data to return
		Server().DataOwnerManager().RequestDataL(pGenericType, writeBuf, finished);
		
		CleanupStack::PopAndDestroy(pGenericType);
		
		Server().GSHInterface().Header(Server().GlobalSharedHeap()).iFinished = finished;

		Server().GSHInterface().Header(Server().GlobalSharedHeap()).SetLockedFlag(EFalse);
        __LOG2("CSBESession::RequestDataSyncL() - Server-side data length: %d, address: 0x%08x", writeBuf.Length(), writeBuf.Ptr());
        //__LOGDATA("CSBESession::RequestDataSyncL()        - %S", writeBuf.Ptr(), writeBuf.Length());

		aMessage.Complete(KErrNone);
        __LOG("CSBESession::RequestDataSyncL() - END");
		}
	
	void CSBESession::SupplyDataSyncL(const RMessage2& aMessage)
	/**
	Handle the client's synchronous request to supply data to the SBE
	*/
		{
        __LOG("CSBESession::SupplyDataSyncL() - START");
		TBool finished = aMessage.Int0();

		CSBGenericTransferType* pGenericType = CSBGenericTransferType::NewL(
			Server().GSHInterface().Header(Server().GlobalSharedHeap())
			.GenericTransferTypeBuffer());
		CleanupStack::PushL(pGenericType);

		TPtrC8& readBuf = Server().GSHInterface().ReadBufferL(Server().GlobalSharedHeap());

		Server().GSHInterface().Header(Server().GlobalSharedHeap()).SetLockedFlag(ETrue);
        __LOG2("CSBESession::SupplyDataSyncL() - Server-side data length: %d, address: 0x%08x", readBuf.Length(), readBuf.Ptr());
        //__LOGDATA("CSBESession::SupplyDataSyncL()         - %S", readBuf.Ptr(), readBuf.Length());

		Server().DataOwnerManager().SupplyDataL(pGenericType, readBuf, finished);

		
		CleanupStack::PopAndDestroy(pGenericType);

		Server().GSHInterface().Header(Server().GlobalSharedHeap()).SetLockedFlag(EFalse);

		aMessage.Complete(KErrNone);
        __LOG("CSBESession::SupplyDataSyncL() - END");
		}
		
	void CSBESession::GetExpectedDataSizeL(const RMessage2& aMessage)
	/**
	Return the size of data that the DOM expects to send back to the PC
	*/
		{
		TUint size;
		TPckg<TUint> sizePkg(size);

		TInt bufferLen = aMessage.GetDesLengthL(0);
		HBufC8* transTypeBuf = HBufC8::NewL(bufferLen);
		CleanupStack::PushL(transTypeBuf);

		TPtr8 transTypeBufDes(transTypeBuf->Des());
		aMessage.ReadL(0, transTypeBufDes);

		CSBGenericTransferType* pGenericType = CSBGenericTransferType::NewL(transTypeBufDes);
		CleanupStack::PopAndDestroy(transTypeBuf);
		CleanupStack::PushL(pGenericType);
		
		aMessage.ReadL(1, sizePkg);
		
		Server().DataOwnerManager().GetExpectedDataSizeL(pGenericType, sizePkg());
		CleanupStack::PopAndDestroy(pGenericType);	
	
		aMessage.WriteL(1, sizePkg);

		aMessage.Complete(KErrNone);
		}
		
	void CSBESession::AllSnapshotsSuppliedL(const RMessage2& aMessage)
	/**
	*/
		{
		Server().DataOwnerManager().AllSnapshotsSuppliedL();

		aMessage.Complete(KErrNone);
		}
		
	void CSBESession::AllSystemFilesRestoredL()
	/**
	Sent by the client to indicate that all registration files 
	@param aMessage IPC message sent from the client
	*/
		{
		Server().DataOwnerManager().AllSystemFilesRestoredL();
		}

	void CSBESession::PrepLargePublicFileListL(const RMessage2& aMessage)
	/**
	Compile a list of public files owned by a particular SID to be backed up
	@param aMessage The message sent by the client to the server
	*/
		{
		HBufC8* pGenericDataTypeBuffer = HBufC8::NewLC(aMessage.GetDesLengthL(1));		
		TPtr8 genericDataTypeBuffer(pGenericDataTypeBuffer->Des());
		
		TDriveNumber drive = static_cast<TDriveNumber>(aMessage.Int0());
		aMessage.ReadL(1, genericDataTypeBuffer);
		TInt cursor = aMessage.Int2();
		TInt maxSize = aMessage.Int3();
		
		CSBGenericDataType* pGenericDataType = CSBGenericDataType::NewL(*pGenericDataTypeBuffer);
		CleanupStack::PopAndDestroy(pGenericDataTypeBuffer);
		CleanupStack::PushL(pGenericDataType);
		
		// Create a transfer type (ignored by the client side, just here for future reference
		CSBGenericTransferType* transType = NULL;
		
		switch (pGenericDataType->DerivedTypeL())
			{
			case ESIDDerivedType:
				{
				
				CSBSecureId* sid = CSBSecureId::NewL(pGenericDataType);
				CleanupStack::PushL(sid);
				transType = CSBSIDTransferType::NewL(sid->SecureIdL(), drive, EPublicFileListing);
				CleanupStack::PopAndDestroy(sid);
				break;
				}
			case EPackageDerivedType:
				{
				CSBPackageId* pkg = CSBPackageId::NewL(pGenericDataType);
				CleanupStack::PushL(pkg);
				transType = CSBPackageTransferType::NewL(pkg->PackageIdL(), drive, ESystemData);
				CleanupStack::PopAndDestroy(pkg);
				break;
				}
			case EJavaDerivedType:
				{
				CSBJavaId* java = CSBJavaId::NewL(pGenericDataType);
				CleanupStack::PushL(java);
				transType = CSBJavaTransferType::NewL(java->SuiteHashL(), drive, EJavaMIDletData);
				CleanupStack::PopAndDestroy(java);
				break;
				}
			default:
				{
				__LOG1("Unknown generic data type supplied, leaving with KErrUnknown (%d)", KErrUnknown);
				User::Leave(KErrUnknown);
				}
			}
		CleanupStack::PushL(transType);
		
		// Get a writeable buffer from the global shared heap
		TPtr8& writeBuf = Server().GSHInterface().WriteBufferL(Server().GlobalSharedHeap());
		Server().GSHInterface().Header(Server().GlobalSharedHeap()).SetLockedFlag(ETrue);
		Server().GSHInterface().Header(Server().GlobalSharedHeap()).GenericTransferTypeBuffer() = transType->Externalise();

		// This is required to look up the MIME type of the file entry and convert it into a textual representation	
		RApaLsSession apaSession;
		User::LeaveIfError(apaSession.Connect());
		CleanupClosePushL(apaSession);

		// Copy file entries from the cursor until either the end of the list is reached or the 
		// max response size ceiling is reached. Store the number added for the call to ReturnLargePublicFileListL
		iFinished = EFalse;
		TInt currentIndex = cursor;
		TInt currentSize = 0;
		iTotalEntries = 0;

		if (currentIndex == 0)
			{
			// Starting from the beginning, remove previously saved entry
			delete iExtCurEntry;
			iExtCurEntry = 0;
			}

		while (currentSize < maxSize)
			{
			if (!iExtCurEntry)
				{
				TEntry entry;
				Server().DataOwnerManager().GetNextPublicFileL(pGenericDataType, (currentIndex == 0), drive, entry);
				if (entry.iName.Length())
					{
					++currentIndex;
								
					// Pull the next entry out of the list and translate into CSBEFileEntry
					CSBEFileEntry* pEntry = CSBEFileEntry::NewLC(entry, apaSession);
					
					// Pack it into the GSH
					iExtCurEntry = pEntry->ExternaliseLC();
					CleanupStack::Pop(iExtCurEntry);
		
					CleanupStack::PopAndDestroy(pEntry);
					}
				}
			// else - use the remaining entry from the previous call
			
			if (iExtCurEntry)
				{
				// If this entry isn't going to bust our max RAM parameter, then pack it in
				currentSize += iExtCurEntry->Size();
				if (currentSize <= maxSize)
					{
					writeBuf.Append(*iExtCurEntry);
					delete iExtCurEntry;
					iExtCurEntry = 0;
					++iTotalEntries;
					}
				// else - leave iExtCurEntry until the next call
				}
			else
				{
				// No more entries left, must have reached the end of the list
				iFinished = ETrue;
				break;					
				}
			}
		CleanupStack::PopAndDestroy(&apaSession);
        __LOG2("CSBESession::PrepLargePublicFileListL() - Server-side data length: %d, address: 0x%08x", writeBuf.Length(), writeBuf.Ptr());
		
		Server().GSHInterface().Header(Server().GlobalSharedHeap()).iFinished = iFinished;

		Server().GSHInterface().Header(Server().GlobalSharedHeap()).SetLockedFlag(EFalse);

		CleanupStack::PopAndDestroy(transType);
		CleanupStack::PopAndDestroy(pGenericDataType);

		aMessage.Complete(KErrNone);
		}
		
	void CSBESession::ReturnLargePublicFileListL(const RMessage2& aMessage)
	/** Return the previously populated buffer to the client
	@param aMessage The message sent by the client to the server */
		{
		TPckg<TBool> finPkg(iFinished);
		TPckg<TInt> totalPkg(iTotalEntries);
		aMessage.WriteL(0, finPkg);
		aMessage.WriteL(1, totalPkg);
		aMessage.Complete(KErrNone);
		}
				
	void CSBESession::ResetTransferBuf()
	/**
	*/
		{
		if (iTransferBuf != NULL)
			{
			delete iTransferBuf;
			iTransferBuf = NULL;
			}

		if (iTransferTextBuf != NULL)
			{
			delete iTransferTextBuf;
			iTransferTextBuf = NULL;	
			}
			
		iArrayCount = 0;
		}
	}