backupandrestore/backupengine/src/sbeconfig.cpp
author hgs
Fri, 15 Oct 2010 15:05:57 +0800
changeset 54 4dc88a4ac6f4
parent 49 c20dd21d1eb4
permissions -rw-r--r--
201041_02

// 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 sbeconfig
// 
//

/**
 @file
*/
#include <e32std.h>
#include "sbepanic.h"
#include "sbeconfig.h"
#include <xml/parser.h>
#include "OstTraceDefinitions.h"
#include "sbtrace.h"
#ifdef OST_TRACE_COMPILER_IN_USE
#include "sbeconfigTraces.h"
#endif

namespace conn
	{
	const TInt KSIDLength = 8;
	
	// XML type
	_LIT8(KMimeType, "text/xml");
	
	// elements
	_LIT8(KConfig, "sbe_config");
	_LIT8(KHeap, "heap");
	_LIT8(KCentRep, "central_repository");
	_LIT8(KDrives, "exclude_drives");
	_LIT8(KAppCloseDelay, "app_close_delay");
	
	_LIT8(KSize, "size");
	_LIT8(KUid, "uid");
	_LIT8(KList, "list");
	
	_LIT8(KRedFact, "reduction_factor");
	_LIT8(KMaxRetries, "max_retries");
	
	_LIT8(KDelay, "delay");
	
	// default setting if no file found
	const TInt KSBEGSHDefaultSize = 2097152;
	const TInt KSBEGSHReductionFactor = 2;
	const TInt KSBEGSHMaxRetries = 5;
	
	const TInt KMinHeapSize = 131072;
	
	const TInt KDefaultDelay = 0;
	
	_LIT_SECURE_ID(KCentRepSID,0x10202BE9);
	_LIT(KConfigFile, "sbeconfig.xml");
	
	/**
	Symbian Constructor
	@param RFs& reference to RFs
	@return CSBEConfig* pointer to CSBEConfig
	*/
	CSBEConfig* CSBEConfig::NewL(RFs& aRFs)
		{
		OstTraceFunctionEntry0( CSBECONFIG_NEWL_ENTRY );
		CSBEConfig* self = new (ELeave) CSBEConfig(aRFs);
		OstTraceFunctionExit0( CSBECONFIG_NEWL_EXIT );
		return self;
		}
	
	/**
	C++ Constructor
	*/
	CSBEConfig::CSBEConfig(RFs& aRFs) : iRFs(aRFs), iFileName(KConfigFile), iConfigTagVisited(EFalse)
		{
		OstTraceFunctionEntry0( CSBECONFIG_CSBECONFIG_CONS_ENTRY );
		SetDefault();
		OstTraceFunctionExit0( CSBECONFIG_CSBECONFIG_CONS_EXIT );
		}
	/** 
	Destructor
	*/
	CSBEConfig::~CSBEConfig()
		{
		OstTraceFunctionEntry0( CSBECONFIG_CSBECONFIG_DES_ENTRY );
		delete iConverter;
		OstTraceFunctionExit0( CSBECONFIG_CSBECONFIG_DES_EXIT );
		}
	
	/**
	Heap Values
	@param TInt& aMaxSize of the heap to try to allocate
	@param TInt& aReductionFactor in case allocation fail
	@param TInt& number of retries to try to reduce the heap by the ReductionFactor
	*/
	void CSBEConfig::HeapValues(TInt& aMaxSize, TInt& aReductionFactor, TInt& aMaxRetries) const
		{
		OstTraceFunctionEntry0( CSBECONFIG_HEAPVALUES_ENTRY );
		aMaxSize = iSBEGSHMaxSize;
		aReductionFactor = iReductionFactor;
		aMaxRetries = iMaxRetries;
		OstTraceFunctionExit0( CSBECONFIG_HEAPVALUES_EXIT );
		}
		
	/**
	Secure Id for central repository, needed deprecated use of centrep tag in xml
	@return TSecureId& aSecureId
	*/	
	TSecureId CSBEConfig::CentRepId() const
		{
		return iCentRepId;
		}
	
	/**
	Exclude list of drives from backup/restore
	@return TDriveList& aDriveList
	*/	
	const TDriveList& CSBEConfig::ExcludeDriveList() const
		{
		return iDrives;
		}
	
	/**
	Extra time delay to close all non-system apps 
	@return TInt& iAppCloseDelay
	*/
	TUint CSBEConfig::AppCloseDelay() const
		{
		return iAppCloseDelay;
		}

	/**
	Set the values to Defaults
	*/	
	void CSBEConfig::SetDefault()
		{
		OstTraceFunctionEntry0( CSBECONFIG_SETDEFAULT_ENTRY );
		iSBEGSHMaxSize = KSBEGSHDefaultSize;
		iCentRepId = KCentRepSID;
		iDrives.SetLength(KMaxDrives);
		iDrives.FillZ();
		iDrives[EDriveZ] = ETrue;
		iReductionFactor = KSBEGSHReductionFactor;
		iMaxRetries = KSBEGSHMaxRetries;
		iAppCloseDelay = KDefaultDelay;
		OstTraceFunctionExit0( CSBECONFIG_SETDEFAULT_EXIT );
		}
		
	/**
	Method to convert string of drives (eg. cdez) to member variable TDriveList
	@param const TDesC8& reference to string
	*/	
	TInt CSBEConfig::StringToDrives(const TDesC8& aDes)
		{
		OstTraceFunctionEntry0( CSBECONFIG_STRINGTODRIVES_ENTRY );
		iDrives.SetLength(KMaxDrives);
		iDrives.FillZ();
		
		TInt err = KErrNone;
		TInt length = aDes.Length();
		for (TInt i = 0; i < length; ++i)
			{
			TInt pos;
			err = iRFs.CharToDrive(aDes.Ptr()[i], pos);
			if (err != KErrNone)
				{
				break;
				}
			iDrives[pos] = ETrue;
			}
		OstTraceFunctionExit0( CSBECONFIG_STRINGTODRIVES_EXIT );
		return err;
		}
	
	/**
	Parses the config file if found
	@leave with System wide Error Codes
	*/	
	void CSBEConfig::ParseL()
		{
		OstTraceFunctionEntry0( CSBECONFIG_PARSEL_ENTRY );
		iRFs.PrivatePath(iFileName);
		TFindFile findFile(iRFs);
		TInt err = findFile.FindByPath(KConfigFile, &iFileName);
		LEAVEIFERROR(err, OstTrace1(TRACE_ERROR, CSBECONFIG_PARSEL, "Leave: %d", err));
		
		iFileName = findFile.File();
		// Connect to the parser
		CParser* parser = CParser::NewLC(KMimeType, *this);
		
		// Parse the file
		Xml::ParseL(*parser, iRFs, iFileName);
		
		CleanupStack::PopAndDestroy(parser);
		OstTraceFunctionExit0( CSBECONFIG_PARSEL_EXIT );
		}
		
	/**
	A method to handle attributes
	@param RAttributeArray& aAttributes 
	@return TInt System Wide Error
	*/	
	TInt CSBEConfig::HandleAttributesElement(const RAttributeArray& aAttributes)
		{
		OstTraceFunctionEntry0( CSBECONFIG_HANDLEATTRIBUTESELEMENT_ENTRY );
		TInt err = KErrNone;
		// Loop through reading out attribute values
		const TUint count = aAttributes.Count();
		for (TInt x = 0; x < count && err == KErrNone; x++)
			{
			TPtrC8 attrib = aAttributes[x].Attribute().LocalName().DesC();
			TPtrC8 value = aAttributes[x].Value().DesC();
			if (!attrib.CompareF(KDelay))
				{
				TLex8 lex(value);
				TInt appCloseDelay = 0;
				err = lex.Val(appCloseDelay);
				if (appCloseDelay < 0)
					{
				    OstTrace0(TRACE_NORMAL, CSBECONFIG_HANDLEATTRIBUTESELEMENT, "Configuration Error: the time delay is negative");
					err = KErrCorrupt;
					}
				else
					{
					iAppCloseDelay = appCloseDelay;
					}
				}
			if (!attrib.CompareF(KRedFact))
				{
				TLex8 lex(value);
				err = lex.Val(iReductionFactor);
				if (iReductionFactor < 0)
					{
				    OstTrace0(TRACE_NORMAL, DUP1_CSBECONFIG_HANDLEATTRIBUTESELEMENT, "Configuration Error: the reductionFactor is negative");
					err = KErrCorrupt;
					}
				}
			else if (!attrib.CompareF(KMaxRetries))
				{
				TLex8 lex(value);
				err = lex.Val(iMaxRetries);
				if (iMaxRetries < 0)
					{
				    OstTrace0(TRACE_NORMAL, DUP2_CSBECONFIG_HANDLEATTRIBUTESELEMENT, "Configuration Error: the maxRetries is negative");
					err = KErrCorrupt;
					}
				}
			if (!attrib.CompareF(KSize))
				{
				TLex8 lex(value);
				err = lex.Val(iSBEGSHMaxSize);
				if (iSBEGSHMaxSize < KMinHeapSize)
					{
				    OstTrace1(TRACE_NORMAL, DUP3_CSBECONFIG_HANDLEATTRIBUTESELEMENT, "Configuration Error: heap size is less then minimum %d", KMinHeapSize);
					err = KErrCorrupt;
					}
				} // if
			else if (!attrib.CompareF(KUid))
				{
				TLex8 lex;
				if (value.Length() >= KSIDLength)
					{
					lex = value.Right(KSIDLength);
					err = lex.Val(iCentRepId.iId, EHex);
					if (iCentRepId.iId == 0)
						{
						err = KErrCorrupt;
						}
					}
				if (err != KErrNone)
					{
				    OstTrace0(TRACE_NORMAL, DUP4_CSBECONFIG_HANDLEATTRIBUTESELEMENT, "Configuration Error: central_repostiory is NOT a HEX number");
					err = KErrCorrupt;
					}
				} // else if
			else if (!attrib.CompareF(KList))
				{
				err = StringToDrives(value);
				if (err != KErrNone)
					{
				    OstTrace0(TRACE_NORMAL, DUP5_CSBECONFIG_HANDLEATTRIBUTESELEMENT, "Configuration Error: list doesn't have valid characters from a-z");
					}
				} // else if
				
			} // for x
		OstTraceFunctionExit0( CSBECONFIG_HANDLEATTRIBUTESELEMENT_EXIT );
		return err;
		}
		
// From MContentHandler
	void CSBEConfig::OnStartDocumentL(const RDocumentParameters& /*aDocParam*/, TInt /*aErrorCode*/)
	/**
	Start of the document, creates Character Converter to convert from/to unicode
	
	@see MContentHandler::OnStartDocumentL()
	@leave if fails to set encoding
	*/
		{
		OstTraceFunctionEntry0( CSBECONFIG_ONSTARTDOCUMENTL_ENTRY );
		// Create a converter for converting strings to Unicode
		iConverter = CCnvCharacterSetConverter::NewL();

		// We only convert from UTF-8 to UTF-16
		if (iConverter->PrepareToConvertToOrFromL(KCharacterSetIdentifierUtf8, iRFs) == CCnvCharacterSetConverter::ENotAvailable)
			{
		    OstTrace0(TRACE_ERROR, CSBECONFIG_ONSTARTDOCUMENTL, "Leave: KErrNotFound");
			User::Leave(KErrNotFound);
			}
		OstTraceFunctionExit0( CSBECONFIG_ONSTARTDOCUMENTL_EXIT );
		}
		
	void CSBEConfig::OnEndDocumentL(TInt /*aErrorCode*/)
	/**
	End of document. destroys converter object
	
	@see MContentHandler::OnEndDocumentL()
	*/
		{
		OstTraceFunctionEntry0( CSBECONFIG_ONENDDOCUMENTL_ENTRY );
		// We've finished parsing the document, hence destroy the converter object
		delete iConverter;
		iConverter = NULL;
		OstTraceFunctionExit0( CSBECONFIG_ONENDDOCUMENTL_EXIT );
		}
		
	void CSBEConfig::OnStartElementL(const RTagInfo& aElement, const RAttributeArray& aAttributes, TInt /*aErrCode*/)
	/**
	Element to parse on the start
	
	@see MContentHandler::OnStartElementL()
	
	@param aElement RTagInfo&
	@param aAttributes RAttributeArray&
	*/
		{
		OstTraceFunctionEntry0( CSBECONFIG_ONSTARTELEMENTL_ENTRY );
		TInt err = KErrNone;
		TPtrC8 localName(aElement.LocalName().DesC());
		if (!localName.CompareF(KConfig))
			{
			iConfigTagVisited = ETrue;
			} // if
		else if (iConfigTagVisited)
			{
			if (!localName.CompareF(KHeap) || !localName.CompareF(KCentRep) || !localName.CompareF(KDrives) || !localName.CompareF(KAppCloseDelay))
				{
				err = HandleAttributesElement(aAttributes);
				} // if
			else
				{
				err = KErrCorrupt;
				} // else if
			} // else if
		else
			{
			err = KErrCorrupt;
			}
		LEAVEIFERROR(err, OstTrace1(TRACE_ERROR, CSBECONFIG_ONSTARTELEMENTL, "Leave: %d", err));
		OstTraceFunctionExit0( CSBECONFIG_ONSTARTELEMENTL_EXIT );
		}
		
	void CSBEConfig::OnEndElementL(const RTagInfo& /*aElement*/, TInt /*aErrorCode*/)
	/**
	Element to parse at the end
	
	@see MContentHandler::OnEndElementL()
	@param const aElement RTagInfo&
	*/
		{
		}
		
	void CSBEConfig::OnContentL(const TDesC8& /*aBytes*/, TInt /*aErrorCode*/)
	/** 
	@see MContentHandler::OnContentL()
	*/
		{
		}
		
	void CSBEConfig::OnStartPrefixMappingL(const RString& /*aPrefix*/, const RString& /*aUri*/, TInt /*aErrorCode*/)
	/** 
	@see MContentHandler::OnStartPrefixMappingL()
	*/
		{
		}
		
	void CSBEConfig::OnEndPrefixMappingL(const RString& /*aPrefix*/, TInt /*aErrorCode*/)
	/** 
	@see MContentHandler::OnEndPrefixMappingL()
	*/
		{
		}
		
	void CSBEConfig::OnIgnorableWhiteSpaceL(const TDesC8& /*aBytes*/, TInt /*aErrorCode*/)
	/** 
	@see MContentHandler::OnIgnorableWhiteSpaceL()
	*/
		{
		}
		
	void CSBEConfig::OnSkippedEntityL(const RString& /*aName*/, TInt /*aErrorCode*/)
	/** 
	@see MContentHandler::OnSkippedEntityL()
	*/
		{
		}
		
	void CSBEConfig::OnProcessingInstructionL(const TDesC8& /*aTarget*/, const TDesC8& /*aData*/, TInt /*aErrorCode*/)
	/** 
	@see MContentHandler::OnProcessingInstructionL()
	*/
		{
		}
		
	void CSBEConfig::OnError(TInt /*aErrorCode*/)
	/** 
	@see MContentHandler::OnError()
	*/
		{
		}
		
	TAny* CSBEConfig::GetExtendedInterface(const TInt32 /*aUid*/)
	/** 
	@see MContentHandler::GetExtendedInterface()
	*/
		{
		return NULL;
		}
		
	
	}