homescreenpluginsrv/hspsmanager/src/hspsserverutil.cpp
branchRCL_3
changeset 114 a5a39a295112
child 118 8baec10861af
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/homescreenpluginsrv/hspsmanager/src/hspsserverutil.cpp	Wed Sep 01 12:22:09 2010 +0100
@@ -0,0 +1,2252 @@
+/*
+* Copyright (c) 2008 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:  Utility for common code.
+*
+*/
+
+
+#include "hspsthememanagement.h"
+#include "hspsserverutil.h"
+#include "hspsodt.h"
+#include "hspsdomdepthiterator.h"
+#include "hspsdomnode.h"
+#include "hspsdomdocument.h"
+#include "hspsconfiguration.h"
+#include "hspsdomlist.h"
+#include "hspsresource.h"
+#include "hspsdomattribute.h"
+#include "hspsmanifest.h"
+#include "bautils.h" 
+#include "sysutil.h"
+#include <syslangutil.h>
+#include <driveinfo.h>
+
+
+_LIT(KHspsFolder, "\\200159c0\\themes\\" );
+_LIT(KSourcesFolder, "\\sources\\");
+_LIT( KThemesFolder, "\\themes\\" );
+_LIT( KDoubleBackSlash, "\\" );
+_LIT8( KHexPrefix8, "0x" );
+_LIT( KClientSources, "c:\\private\\%x\\%D\\%D\\%D\\%S\\sources\\%S" );
+
+// -----------------------------------------------------------------------------
+// hspsServerUtil::GenerateConfigurationAttributesL
+// -----------------------------------------------------------------------------
+//
+void hspsServerUtil::GenerateConfigurationAttributesL( ChspsODT& aOdt )
+    {
+    ChspsDomDocument& dom = aOdt.DomDocument();
+    ChspsDomDepthIterator* iter = ChspsDomDepthIterator::NewL( *dom.RootNode() );   
+    CleanupStack::PushL( iter );
+                        
+    ChspsDomNode* prevNode = NULL;
+    ChspsDomNode* node = iter->First();
+    while( node && prevNode != node )
+        {           
+        const TDesC8& name = node->Name();
+        
+        // Configuration element 
+        if ( name == KConfigurationElement )
+            {                                                   
+            // Add attributes for the configuration node        
+            // NOTE! ID attribute is not set here           
+            TPtrC8 ptr = KManifestTypeApp().Ptr();
+            switch ( aOdt.ConfigurationType() )
+                {
+                case EhspsViewConfiguration:
+                    {
+                    ptr.Set( KManifestTypeView );
+                    }
+                    break;
+                case EhspsWidgetConfiguration:
+                    {
+                    ptr.Set( KManifestTypeWidget );
+                    }
+                    break;
+                case EhspsTemplateConfiguration:
+                    {
+                    ptr.Set( KManifestTypeTemplate );
+                    }
+                    break;
+                default:
+                    {                   
+                    }
+                }
+                    
+            AddAttributeDescL( *node, KConfigurationAttrType, ptr );
+            
+            AddAttributeNumericL( *node, KConfigurationAttrInterface, aOdt.RootUid(), EHex );
+            
+            AddAttributeNumericL( *node, KConfigurationAttrUid, aOdt.ThemeUid(), EHex );
+            
+            // Create "name" and "_name" attributes, of which latter holds the entity reference 
+            // (logical key for finding localizad strings)
+            HBufC8* nameBuf = HBufC8::NewLC( aOdt.ThemeFullName().Length() );
+            TPtr8 namePtr( nameBuf->Des() );
+            namePtr.Copy( aOdt.ThemeFullName() );
+            AddAttributeDescL( *node, KConfigurationAttrName, namePtr ); // will be updated when localized
+            AddAttributeDescL( *node, KConfigurationAttrNameEntity, namePtr ); // logical id
+            CleanupStack::PopAndDestroy( nameBuf );                        
+
+            // Create theme version attribute
+            HBufC8* tv = HBufC8::NewLC( aOdt.ThemeVersion().Length() );
+            TPtr8 tvPtr( tv->Des() );
+            tvPtr.Copy( aOdt.ThemeVersion() );
+            AddAttributeDescL( *node, KConfigurationAttrVersion, tvPtr );
+            CleanupStack::PopAndDestroy( tv );
+            
+            AddAttributeNumericL ( *node, KConfigurationAttrMultiInstance,
+                aOdt.MultiInstance(), EDecimal );
+
+            AddAttributeDescL( *node, KConfigurationAttrState, KConfStateNotConfirmed );
+            
+            const TInt descLength = aOdt.Description().Length();
+            if ( descLength )
+                {
+                // Add description of the widget (16 > 8bit conversion) 
+                HBufC8* buf = HBufC8::NewLC( descLength );
+                TPtr8 bufPtr( buf->Des() );
+                bufPtr.Copy( aOdt.Description() );
+                AddAttributeDescL( *node, KConfigurationAttrDesc, bufPtr ); // will be updated when localized
+                AddAttributeDescL( *node, KConfigurationAttrDescEntity, bufPtr ); // logical id
+                CleanupStack::PopAndDestroy( buf );
+                }
+            else
+                {
+                AddAttributeDescL( *node, KConfigurationAttrDesc, KNullDesC8 );
+                }
+            
+            }       
+        
+        prevNode = node;    
+        node = iter->NextL();
+        }
+    CleanupStack::PopAndDestroy( iter );
+    }
+
+// -----------------------------------------------------------------------------
+// ChspsInstallationHandler::GenerateObjectAttributesL()
+// -----------------------------------------------------------------------------
+//
+void hspsServerUtil::GenerateObjectAttributesL( ChspsODT& aOdt )
+    {
+    // Find the configuration node
+    ChspsDomDocument& dom = aOdt.DomDocument();     
+    ChspsDomNode* configurationNode = dom.RootNode();
+    if ( !configurationNode )
+        {
+#ifdef _hsps_DEBUG_       
+        RDebug::Print( _L("hspsServerUtil::GenerateObjectAttributesL(): - invalid ODT!") );
+#endif      
+        User::Leave( KErrGeneral );
+        }
+    
+    // Find resources node from the XML definition (child of the configuration node)
+    ChspsDomList& childsList = configurationNode->ChildNodes(); 
+    ChspsDomNode* resourcesNode = (ChspsDomNode *)childsList.FindByName( KResourcesElement );
+    TInt popResources = EFalse;
+    if ( !resourcesNode )
+        {
+        // Create a node and add it into the configuration node
+        resourcesNode = dom.CreateElementNSL( 
+            KResourcesElement,      
+            configurationNode->Namespace()          
+            );                              
+        CleanupStack::PushL( resourcesNode );
+        popResources = ETrue;
+        configurationNode->AddChildL( resourcesNode );
+        resourcesNode->SetParent( configurationNode );
+        }    
+                              
+    // Loop ODT's resources (parsed ealier from the manifest file)
+    const TInt resourceCount = aOdt.ResourceCount();
+    TBool addResource = EFalse;
+    for( TInt resourceIndex=0; resourceIndex < resourceCount; resourceIndex++ )
+        {                
+        ChspsResource& resource = aOdt.ResourceL( resourceIndex );                                
+        
+        addResource = EFalse;
+        
+        // If resource is located under the sources folder
+        if ( resource.FileName().FindF( KSourcesFolder ) > 0 )        
+            {                        
+            // If resource is for the active device language or it's a locale independent resource
+            addResource = ( resource.Language() == aOdt.OdtLanguage() || resource.Language() == ELangNone );                                     
+            }
+                
+        if ( addResource )        	
+        	{          	
+	         // Create an object node and add the object into the resources list                    
+	        ChspsDomNode* objectNode = dom.CreateElementNSL( 
+	            KObjectElement,         
+	            resourcesNode->Namespace()          
+	            );                              
+	        CleanupStack::PushL( objectNode );
+	        resourcesNode->AddChildL( objectNode );
+	        objectNode->SetParent( resourcesNode );
+	        
+	        // Name (16->8bit conversion)
+	        HBufC8* nameBuf = HBufC8::NewLC( resource.ResourceId().Length() );
+	        TPtr8 namePtr( nameBuf->Des() );
+	        namePtr.Copy( GetFixedOdtName( resource.ResourceId() ) );            
+	        
+	        AddAttributeDescL( *objectNode, KObjectAttrFilename, namePtr );
+	        CleanupStack::PopAndDestroy( nameBuf );
+	        
+	        // Media type
+	        TPtrC8 mimePtr( resource.MimeType().Des8() );           
+	        if ( mimePtr.Length() )
+	            {
+	            AddAttributeDescL( *objectNode, KObjectAttrMediatype, mimePtr );
+	            }
+	        
+	        // Tag
+            TPtrC tagPtr( resource.Tags() );           
+            if ( tagPtr.Length() )
+                {
+                // (16->8bit conversion)
+                HBufC8* buf = HBufC8::NewLC( tagPtr.Length() );
+                TPtr8 bufPtr( buf->Des() );
+                bufPtr.Copy( tagPtr );
+                AddAttributeDescL( *objectNode, KObjectAttrTag, bufPtr );
+                CleanupStack::PopAndDestroy( buf );
+                }
+	                
+	        // Path 
+	        TInt pos = resource.FileName().FindF( KHspsFolder );
+	        if ( pos > 0 )
+	            {
+	            // Remove filename and extension from the path
+	            TParsePtrC parserPtr( resource.FileName() );
+	            TFileName path( parserPtr.DriveAndPath() );
+	            	            
+	            // Remove path to the Definition repository 
+	            path.Copy( path.Mid( pos + KHspsFolder().Length() ) );
+	            
+	            // Fix path references for localized resources
+	            GetLocaleIndependentResourcePath( resource.Language(), path );	            	            
+	            
+	            // 16->8bit conversion	            
+	            HBufC8 *pathBuf = HBufC8::NewLC( path.Length() );                  	            
+	            pathBuf->Des().Copy( path );
+	            	            	            
+	            AddAttributeDescL( *objectNode, KObjectAttrPath, pathBuf->Des() );
+	            CleanupStack::PopAndDestroy( pathBuf );
+	            }
+	        
+	        // Now the document has an ownership of the objectnode
+	        CleanupStack::Pop( objectNode );
+	        
+        	}
+        
+        } // for loop
+
+    if ( popResources )
+        {
+        // Now the document has an ownership of the resourcesNode
+        CleanupStack::Pop( resourcesNode );
+        }
+    }
+
+
+TFileName hspsServerUtil::GetFixedOdtName( 
+        const TDesC& aNameAndExtension )
+    {        
+    TParsePtrC parsePtr( aNameAndExtension );                      
+    TPtrC fileExtension = parsePtr.Ext();
+    if ( fileExtension.Length() > 2 && fileExtension.Left(2).CompareF( _L(".o") ) == 0 )
+        {            
+        // Strip the first letter            
+        TInt odtIndex(0);                  
+        TLex lex( fileExtension.Mid(3) );
+        if ( lex.Val( odtIndex ) == KErrNone && odtIndex >= 0 )
+            {                
+            fileExtension.Set( _L(".o0000") );
+            }
+        }
+    
+    TFileName fileName( parsePtr.Name() );
+    fileName.Append( fileExtension );
+    return fileName;
+    }
+
+// -----------------------------------------------------------------------------
+// hspsServerUtil::GetLocaleIndependentResourcePath
+// -----------------------------------------------------------------------------
+//
+void hspsServerUtil::GetLocaleIndependentResourcePath(
+		const TLanguage& aResourceLanguage,		 
+		TFileName& aPath )
+	{
+	TBool isLocaleSpecificResource = ( aResourceLanguage != ELangNone );		
+	
+	// If locale specific resource
+    if ( isLocaleSpecificResource )
+    	{
+    	// Remove locale specific subfolder from the path
+    	TInt pos = aPath.FindF( KSourcesFolder );
+    	if ( pos )
+    		{    		    		    		    	
+    		aPath.Copy( aPath.Left( pos + KSourcesFolder().Length() ) );
+    		}
+    	}    
+	}
+
+// -----------------------------------------------------------------------------
+// hspsServerUtil::GetRelativeResourcePath
+// -----------------------------------------------------------------------------
+//
+void hspsServerUtil::GetRelativeResourcePath(
+        const TFileName& aSourceFile,
+        TPath& aRelativePath )
+    {    
+    // find last part (structure after "/themes/") 
+    TInt pos = aSourceFile.FindF( KThemesFolder );
+    if( pos != KErrNotFound )
+        {
+        pos += KThemesFolder().Length();        
+        TInt len( aSourceFile.Length() - pos );               
+        aRelativePath.Copy( aSourceFile.Right( len ) );                     
+        }    
+    }
+
+// -----------------------------------------------------------------------------
+// hspsServerUtil::AddAttributeNumericL
+// -----------------------------------------------------------------------------
+//
+TInt hspsServerUtil::AddAttributeNumericL(
+        ChspsDomNode& aNode,
+        const TDesC8& aAttrName, 
+        const TInt aValue,
+        const TRadix aFormat )
+    {
+    // Format change
+    // Value string, 10 -> Max decimal 4294967295 , max hexadecimal 0xXXXXXXXX
+    const TInt KMaxLength = 10;
+    TBuf8<KMaxLength> attValueDes8; 
+    
+    if ( aFormat == EHex )
+        {
+        _LIT8( KFormat8, "%X" );
+        _LIT8( KHexPrefix, "0x" );
+        attValueDes8.Append( KHexPrefix );
+        attValueDes8.AppendFormat( KFormat8, aValue );
+        }
+    else // EDecimal
+        {
+        _LIT8( KFormat8, "%d" );
+        attValueDes8.AppendFormat( KFormat8, aValue );
+        }
+    
+    ChspsDomList& attrList = aNode.AttributeList();
+    ChspsDomAttribute* attr = static_cast<ChspsDomAttribute*>( attrList.FindByName(aAttrName) );
+    if ( attr )
+        {               
+        // Replace value of the attribute
+        attr->SetValueL( attValueDes8 );
+        }
+    else
+        {              
+        // Add an attribute
+        ChspsDomAttribute* attr = ChspsDomAttribute::NewL( aAttrName, aNode.StringPool() );    
+        CleanupStack::PushL( attr );            
+        attr->SetValueL( attValueDes8 );
+        ChspsDomList& attrList = aNode.AttributeList();
+        attrList.AddItemL( attr );    //takes ownership
+        CleanupStack::Pop( attr );
+        }
+    
+    return KErrNone;
+    }
+
+// -----------------------------------------------------------------------------
+// hspsServerUtil::AddAttributeDescL
+// -----------------------------------------------------------------------------
+//
+void hspsServerUtil::AddAttributeDescL(
+        ChspsDomNode& aNode,
+        const TDesC8& aAttrName, 
+        const TDesC8& aValue )
+    {
+    ChspsDomList& attrList = aNode.AttributeList();
+    ChspsDomAttribute* attr = static_cast<ChspsDomAttribute*>( attrList.FindByName(aAttrName) );
+    if ( attr )
+        {               
+        // Replace value of the attribute
+        attr->SetValueL( aValue );
+        }
+    else
+        {              
+        // Add an attribute
+        ChspsDomAttribute* attr = ChspsDomAttribute::NewL( aAttrName, aNode.StringPool() );    
+        CleanupStack::PushL( attr );            
+        attr->SetValueL( aValue );
+        ChspsDomList& attrList = aNode.AttributeList();
+        attrList.AddItemL( attr );    //takes ownership
+        CleanupStack::Pop( attr );
+        }        
+    }
+
+
+// -----------------------------------------------------------------------------
+// Finds a configuration node with the provided id attribute
+// -----------------------------------------------------------------------------
+//
+ChspsDomNode* hspsServerUtil::FindConfigurationNodeL(
+		ChspsODT& aOdt,		
+		const TInt aConfigurationId )
+	{
+	__UHEAP_MARK;
+	
+    ChspsDomNode *configurationNode = NULL;
+    
+    ChspsDomDocument& dom = aOdt.DomDocument();
+    ChspsDomDepthIterator* iter = ChspsDomDepthIterator::NewL( *dom.RootNode() );
+    CleanupStack::PushL( iter );
+
+    // Find a configuration node with an id attribute that matches the provided id
+    ChspsDomNode* node = iter->First();
+    ChspsDomNode* prevNode = NULL;
+    TBool jobDone = EFalse;
+    while( node && !jobDone && prevNode != node )    
+        {                
+        const TDesC8& name = node->Name();
+        
+        // An element was found 
+        if ( name == KConfigurationElement )
+            {           
+            ChspsDomList& attrList = node->AttributeList();                    
+            ChspsDomAttribute* idAttr = static_cast<ChspsDomAttribute*>( attrList.FindByName(KConfigurationAttrId) );            
+            if ( !idAttr )
+                {
+                // Return with NULL
+                jobDone = ETrue;                                                 
+                }
+            else
+                {            
+                TInt id(0);            
+                const TDesC8& idValue = idAttr->Value();                        
+                TLex8 lex( idValue );                        
+                lex.Val( id );            
+                if ( aConfigurationId == id )
+                    {
+                    configurationNode = node;        
+                    jobDone = ETrue;
+                    }
+                }
+            }
+        
+        prevNode = node;        
+        node = iter->NextL();
+        }
+    CleanupStack::PopAndDestroy( iter );
+
+#ifdef _hsps_DEBUG_
+    if ( !configurationNode )
+        {
+        RDebug::Print( _L("hspsServerUtil::FindConfigurationNodeL(): failed to find the configuration node") );
+        }
+#endif
+    
+    __UHEAP_MARKEND;
+    
+    return configurationNode;
+    }
+
+// -----------------------------------------------------------------------------
+// Finds a plugin node with the provided id
+// -----------------------------------------------------------------------------
+//
+ChspsDomNode* hspsServerUtil::FindPluginNodeL(
+        ChspsODT& aOdt,        
+        const TInt aPluginId )
+    {            
+    __UHEAP_MARK;
+    
+    ChspsDomNode* pluginNode = NULL;
+    
+    ChspsDomDocument& dom = aOdt.DomDocument();
+    ChspsDomDepthIterator* iter = ChspsDomDepthIterator::NewL( *dom.RootNode() );
+    CleanupStack::PushL( iter );
+
+    // Find a plugin node with the provided id attribute
+    ChspsDomNode* node = iter->First();
+    ChspsDomNode* prevNode = NULL;
+    TBool jobDone = EFalse;
+    while( node && !jobDone && prevNode != node )
+        {                
+        const TDesC8& name = node->Name();
+        
+        // Plugin element was found 
+        if ( name == KPluginElement )
+            {           
+            ChspsDomList& attrList = node->AttributeList();                    
+            ChspsDomAttribute* idAttr = static_cast<ChspsDomAttribute*>( attrList.FindByName(KPluginAttrId) );            
+            if ( !idAttr )
+                {
+                // Mandatory information is missing for some reason (should be set at installation handler)!
+                // Exit with NULL
+                jobDone = ETrue;                
+                }
+            else
+                {            
+                TInt id(0);            
+                const TDesC8& idValue = idAttr->Value();                        
+                TLex8 lex( idValue );                        
+                lex.Val( id );            
+                if ( aPluginId == id )
+                    {
+                    pluginNode = node;
+                    jobDone = ETrue;
+                    }
+                }
+            }
+        
+        prevNode = node;        
+        node = iter->NextL();        
+        }
+    CleanupStack::PopAndDestroy( iter );
+
+#ifdef _hsps_DEBUG_
+    if ( !pluginNode )
+        {
+        RDebug::Print( _L("hspsServerUtil::FindPluginNodeL(): failed to find the plugin node") );
+        }
+#endif
+    
+    __UHEAP_MARKEND;
+    
+    return pluginNode;
+    }
+
+// -----------------------------------------------------------------------------
+// hspsServerUtil::GetConfigurationNameFromDomL
+// -----------------------------------------------------------------------------
+//
+TPtrC8 hspsServerUtil::FindConfigurationAttrL( 
+    const ChspsODT& aOdt,
+    const TDesC8& aAttr )
+    {
+    TPtrC8 ptr;
+    
+    // Get ODT's DOM and find the 1st configuration node
+    ChspsDomNode* confNode = aOdt.DomDocument().RootNode();
+    if( !confNode || confNode->Name().CompareF( KConfigurationElement) != 0 )
+        {            
+        User::Leave( KErrGeneral );            
+        }                
+        
+    // Find the name attribute and return it's value
+    ChspsDomList& attrList = confNode->AttributeList();
+    ChspsDomAttribute* attr = static_cast<ChspsDomAttribute*>( attrList.FindByName(aAttr) );                
+    if ( !attr )
+        {
+        User::Leave( KErrGeneral );
+        }            
+    ptr.Set( attr->Value() );
+            
+    return ptr;
+    }    
+
+// -----------------------------------------------------------------------------
+// hspsServerUtil::RemoveResourceFilesL
+// -----------------------------------------------------------------------------
+void hspsServerUtil::RemoveResourceFilesL(        
+        CFileMan& aFilemanager,
+        RFs& aFs,
+        const TInt aAppUid,
+        const ChspsODT& aPluginODT )
+    {                           
+    // Remove all plug-in resources
+    for( TInt i=0; i< aPluginODT.ResourceCount(); i++ )
+        {
+        ChspsResource& r = aPluginODT.ResourceL( i );       
+        if( r.FileName().Find( KSourcesFolder ) > 0 )
+            {      
+            TPtrC id( GetFixedOdtName( r.ResourceId() ) );
+            const TDesC& ver( aPluginODT.ThemeVersion() );
+            TFileName resource;
+            resource.Format( 
+                    KClientSources,
+                    aAppUid,
+                    aPluginODT.RootUid(),
+                    aPluginODT.ProviderUid(),
+                    aPluginODT.ThemeUid(),
+                    &ver,
+                    &id );
+            if( BaflUtils::FileExists( aFs, resource ) )
+                {
+                aFilemanager.Delete( resource, 0 );
+                }            
+            }
+        }
+    }
+
+
+// -----------------------------------------------------------------------------
+// hspsServerUtil::CopyResourceFilesL
+// -----------------------------------------------------------------------------
+TInt hspsServerUtil::CopyResourceFilesL(
+        ChspsODT& aAppODT,
+        RFs& aFs,
+        CFileMan& aFilemanager,
+        const TInt aDeviceLanguage,
+        const TInt aConfUid,
+        const TDesC& aDestination,
+        const TBool aIsRelevant )
+    {
+    TInt error( KErrNone );
+    RPointerArray<ChspsResource> widgetResources; // Objects are not owned.
+    CleanupClosePushL( widgetResources );
+    
+    // Find resources for the language or common to all languages or test resources    
+    // if others couldn't be found
+    GetResourcesForLanguageL( 
+        aAppODT,
+        aConfUid,
+        (TLanguage)aDeviceLanguage,
+        widgetResources );
+        
+    // Copy the resources found    
+    for( TInt i = 0; ( i < widgetResources.Count() && !error ); i++ )
+        {      
+        ChspsResource* resource = widgetResources[i];
+        if( !resource )
+            {
+            continue;
+            }
+                
+        // Get relative path under the themes folder 
+        TPath relativePath;
+        GetRelativeResourcePath( 
+                resource->FileName(),
+                relativePath );
+        
+        // Strip language indicator from the relative path                                       
+        GetLocaleIndependentResourcePath( 
+                resource->Language(),                            
+                relativePath );   
+        
+        // Finalize target path
+        TPath targetPath;
+        targetPath.Copy( aDestination );
+        targetPath.Append( relativePath );        
+             
+        // Create target path and copy files when required
+        error = CopyResourceFileL(
+                aFs,
+                aFilemanager, 
+                targetPath,
+                resource->FileName(),
+                aIsRelevant );        
+        if ( error == KErrAlreadyExists )
+            {
+            error = KErrNone;
+            }
+        
+        } // copy loop       
+        
+    widgetResources.Reset();
+    CleanupStack::PopAndDestroy( 1, &widgetResources );
+
+    return error;
+    }  
+
+// -----------------------------------------------------------------------------
+// hspsServerUtil::CopyResourceFileL
+// -----------------------------------------------------------------------------
+//
+TInt hspsServerUtil::CopyResourceFileL(
+        RFs& aFs,
+        CFileMan& aFilemanager,
+        const TPath& aTargetPath,
+        const TFileName& aSourceFile,
+        const TBool aIsRelevant )
+    {
+    // Construct target file with full path.
+    TFileName targetFile;
+    
+    TParse targetParser;
+    targetParser.Set( aTargetPath, NULL, NULL );    
+    
+    if( targetParser.NamePresent() )
+        {
+        targetFile = aTargetPath;
+        }
+    else
+        {
+        TParse sourceParser;
+        sourceParser.Set( aSourceFile, NULL, NULL );
+        targetFile = targetParser.DriveAndPath();
+        targetFile.Append( sourceParser.NameAndExt() );    
+        }
+
+    TInt error = KErrNone;
+    
+    if ( aIsRelevant 
+            || hspsServerUtil::ResourceCopyIsRelevantL( 
+                aSourceFile,
+                targetFile,
+                aFs ) 
+        )
+        {
+        // Make target folder
+        error = aFs.MkDirAll( aTargetPath );
+        if( error == KErrAlreadyExists )
+            {
+            // lets ignore error if directory already exists                
+            error = KErrNone;
+            }
+        
+        if( !error )
+            {        
+            // Slowish operation
+            error = aFilemanager.Copy( 
+                    aSourceFile, 
+                    aTargetPath );
+            }
+            
+        if( !error )
+            {
+            // Clear readonly file attribs that might be inherited from the source file                
+            aFilemanager.Attribs( 
+                aTargetPath,
+                0,
+                KEntryAttReadOnly,
+                TTime( 0 ) ); // TTime(0) = preserve original time stamp.                        
+            }        
+        }                
+    
+    return error;
+    }
+
+// -----------------------------------------------------------------------------
+// hspsServerUtil::ResourceCopyIsRelevant
+// -----------------------------------------------------------------------------
+//
+TBool hspsServerUtil::ResourceCopyIsRelevantL( 
+    const TDesC& aSource,
+    const TDesC& aTarget,
+    RFs& aFs )
+    {
+    // Basic sanity check.
+    if( aSource.Length() == 0 || aTarget.Length() == 0 )
+        {
+        return EFalse;
+        }
+
+    // Collect data from files.
+
+    TEntry targetEntry;
+    TInt entryError = aFs.Entry( aTarget, targetEntry ); 
+    if( entryError == KErrNotFound ||
+        entryError == KErrPathNotFound )
+        {
+        // Target does not exist. Copy needed.
+        return ETrue;
+        }           
+    else if( entryError != KErrNone ) 
+        {
+        // All other errors handled here. Better not to copy.
+        return EFalse;
+        }
+    
+    TEntry sourceEntry;
+    entryError = aFs.Entry( aSource, sourceEntry ); 
+    if( entryError != KErrNone )
+        {
+        // Problem. Do not copy.
+        return EFalse;
+        }            
+    
+    TParse sourceParser;
+    sourceParser.Set( aSource, NULL, NULL );
+    
+    TParse targetParser;
+    targetParser.Set( aTarget, NULL, NULL );
+            
+    // We have tdesc of target drive but SysUtils need TDriveNumber instead
+    // so extract it from tdesc. default to C.
+    TInt targetDriveNumber = EDriveC;
+        
+    // Sanity checks before accessing descriptor (prevent "out of bounds" panic).
+    if( targetParser.DrivePresent() &&
+        targetParser.Drive().Length() > 0 )
+        {
+        // Use tmp variable so that CharToDrive does not mess our fallback
+        // value. (it should not do that in case of error... but better
+        // to do fool-proof.)
+        TInt tmpDriveNumber = EDriveC;
+        
+        // Convert from TDesC to enumeration of drives.
+        if( RFs::CharToDrive( targetParser.Drive()[0], tmpDriveNumber ) == KErrNone )
+            {
+            targetDriveNumber = tmpDriveNumber;
+            }        
+        }   
+            
+    // Size and time stamp identical?
+    if( sourceEntry.iSize == targetEntry.iSize &&
+        sourceEntry.iModified == targetEntry.iModified )
+        {
+        return EFalse;
+        }    
+    
+    // Check required disk space.
+    TInt requiredDiskSpace = 0;    
+    if( sourceEntry.iSize > targetEntry.iSize )
+        {
+        requiredDiskSpace = sourceEntry.iSize - targetEntry.iSize; 
+        }
+    else
+        {
+        requiredDiskSpace = sourceEntry.iSize;
+        }
+    
+    if( requiredDiskSpace != 0 )
+        {
+        if( SysUtil::DiskSpaceBelowCriticalLevelL( &aFs, requiredDiskSpace, targetDriveNumber ) )
+            {
+            return EFalse;
+            }
+        }
+        
+    // All tests passed.
+    return ETrue;
+    }
+
+// -----------------------------------------------------------------------------
+// hspsServerUtil::UpdateConfigurationStateL
+// -----------------------------------------------------------------------------
+//
+void hspsServerUtil::UpdateConfigurationStateL( 
+    ChspsODT& aOdt,
+    TDesC8& aConfState,
+    TDesC8& aNextConfState,
+    TBool& aOdtUpdated )
+    {
+    aOdtUpdated = EFalse;
+    ChspsDomDocument& dom = aOdt.DomDocument();
+    ChspsDomDepthIterator* iter = ChspsDomDepthIterator::NewL( *dom.RootNode() );   
+    CleanupStack::PushL( iter );                  
+    ChspsDomNode* prevNode = NULL;
+    ChspsDomNode* node = iter->First();
+    
+    while( node && prevNode != node )
+        {           
+        const TDesC8& name = node->Name();
+        
+        // Configuration element 
+        if ( name == KConfigurationElement )
+            {
+            ChspsDomList& attrList = node->AttributeList();
+            ChspsDomAttribute* attr = static_cast<ChspsDomAttribute*>( 
+                attrList.FindByName( KConfigurationAttrState ) );               
+            if ( attr && attr->Value().CompareF( aConfState ) == 0 )
+                {
+                attr->SetValueL( aNextConfState );
+                aOdtUpdated = ETrue;
+                }           
+            }       
+        
+        prevNode = node;    
+        node = iter->NextL();
+        }
+
+    CleanupStack::PopAndDestroy( iter );
+    
+    }
+
+// -----------------------------------------------------------------------------
+// hspsServerUtil::UpdateAppConfigurationStateL
+// -----------------------------------------------------------------------------
+//
+void hspsServerUtil::UpdateAppConfigurationStateL( 
+        ChspsODT& aAppOdt,
+        const TDesC8& aConfState,
+        const TDesC8& aNextConfState )
+    {
+    __ASSERT_DEBUG( aConfState.Length() > 0, User::Leave( KErrArgument) );
+    __ASSERT_DEBUG( aNextConfState.Length() > 0, User::Leave( KErrArgument) );
+        
+    // Update application configuration state
+    ChspsDomNode* appConfNode = hspsServerUtil::FindNodeByAttributeL(
+        aAppOdt,
+        KConfigurationElement,
+        KConfigurationAttrType,
+        KConfTypeApp );
+    ChspsDomList& attrList = appConfNode->AttributeList();
+    ChspsDomAttribute* attr = static_cast<ChspsDomAttribute*>( 
+        attrList.FindByName( KConfigurationAttrState ) );               
+    if ( attr && attr->Value().CompareF( aConfState ) == 0 )
+        {
+        attr->SetValueL( aNextConfState );
+        }
+    }
+
+// -----------------------------------------------------------------------------
+// hspsServerUtil::FindNodeByAttributeL
+// -----------------------------------------------------------------------------
+//
+ChspsDomNode* hspsServerUtil::FindNodeByAttributeL( 
+    ChspsODT& aOdt,
+    const TDesC8& aNodeName,
+    const TDesC8& aAttrName, 
+    const TDesC8& aAttrValue)
+    {
+    __UHEAP_MARK;
+    
+    ChspsDomDocument& dom = aOdt.DomDocument();
+    ChspsDomDepthIterator* iter = ChspsDomDepthIterator::NewL( *dom.RootNode() );   
+    CleanupStack::PushL( iter );
+                        
+    ChspsDomNode* foundNode( NULL );
+    ChspsDomNode* prevNode( NULL );
+    ChspsDomNode* node = iter->First();
+    while( node && 
+           prevNode != node &&
+           foundNode == NULL )
+        {           
+        const TDesC8& name = node->Name();        
+        if ( name.CompareF( aNodeName ) == 0 )
+            {
+            // Node name match
+            ChspsDomList& attrList = node->AttributeList();
+            ChspsDomAttribute* attr = static_cast<ChspsDomAttribute*>( 
+                attrList.FindByName( aAttrName ) );               
+            if ( attr && attr->Value().CompareF( aAttrValue ) == 0 )
+                {
+                // Attribute name and value match
+                foundNode = node;
+                }           
+            }       
+        // Get next node
+        prevNode = node;    
+        node = iter->NextL();
+        }
+
+    CleanupStack::PopAndDestroy( iter );
+    
+    __UHEAP_MARKEND;
+    
+    return foundNode;
+    }
+
+// -----------------------------------------------------------------------------
+// hspsServerUtil::FindUniquePluginsL
+// -----------------------------------------------------------------------------
+//
+void hspsServerUtil::FindUniquePluginsL( 
+        ChspsODT& aOdt, 
+        RArray<TInt>& aPluginArray )
+    {    
+    aPluginArray.Reset();          
+    ChspsDomDocument& dom = aOdt.DomDocument();
+    ChspsDomDepthIterator* iter = ChspsDomDepthIterator::NewL( *dom.RootNode() );
+    CleanupStack::PushL( iter );
+    
+    ChspsDomNode* node = iter->First();
+    ChspsDomNode* prevNode = NULL;
+    while( node && prevNode != node )
+       {                
+       const TDesC8& name = node->Name();
+       
+       // Plugin element was found 
+       if ( name == KPluginElement )
+           {           
+           ChspsDomList& attrList = node->AttributeList();                    
+           ChspsDomAttribute* uidAttr = static_cast<ChspsDomAttribute*>( attrList.FindByName(KPluginAttrUid) );            
+           if ( uidAttr )
+               {     
+               // Convert from hex to int
+               const TUid pluginUid = ConvertDescIntoUid( uidAttr->Value() );               
+               if ( pluginUid.iUid > 0 )
+                   {
+                   TBool isUnique = ETrue;
+                   for( TInt i=0; isUnique && i<aPluginArray.Count();i++ )
+                       {
+                       if ( aPluginArray[i] == pluginUid.iUid )
+                           {
+                           isUnique=EFalse;
+                           }
+                       }
+                   if ( isUnique )
+                       {
+                       aPluginArray.Append( pluginUid.iUid );
+                       }
+                   }               
+               }
+           }
+           
+           prevNode = node;        
+           node = iter->NextL();        
+           }
+   CleanupStack::PopAndDestroy( iter );      
+}
+
+//----------------------------------------------------------------------------
+// CHspsServiceUtilities::HexString2Uint
+// ----------------------------------------------------------------------------
+//
+TInt hspsServerUtil::HexString2Uint(
+    const TDesC8& aStr,
+    TUint& aTrg )
+    {
+    // Assign to lexer.
+    TLex8 lex( aStr );
+    
+    // Ignore preceding "0x" if it exists. TLex does not know how to handle that
+    // and returns just zero.
+    if( aStr.Length() >= KHexPrefix8().Length() &&
+        aStr.FindF( KHexPrefix8() ) == 0 )
+        {
+        lex.Inc( KHexPrefix8().Length() );
+        }
+    
+    // Actual conversion.
+    TInt status = KErrNone;
+    TUint val = 0;
+    status = lex.Val( val, EHex );
+    
+    // Error check.
+    if( status == KErrNone )
+        {
+        aTrg = val;
+        }
+    
+    return status;
+    }
+
+//----------------------------------------------------------------------------
+// CHspsServiceUtilities::DecString2Int
+// ----------------------------------------------------------------------------
+//
+TInt hspsServerUtil::DecString2Int(
+    const TDesC8& aStr )
+    {
+    TLex8 lex( aStr );
+    TUint value;
+    lex.Mark();
+    
+    while ( lex.Peek().IsDigit() )
+        {
+        lex.Inc();
+        }
+    TPtrC8 uidToken = lex.MarkedToken();
+    TLex8 uidLex( uidToken );
+    TInt err = uidLex.Val( value, EDecimal );
+    
+    return value;
+    }
+// -----------------------------------------------------------------------------
+// Returns a count of plugin instances.
+// -----------------------------------------------------------------------------
+//
+void hspsServerUtil::PluginInstanceCountL(
+        const ChspsODT& aAppODT,        
+        const TInt aPluginUid,
+        TInt& aInstanceCount )              
+    {       
+    aInstanceCount = 0;
+    
+    ChspsDomDocument& dom = aAppODT.DomDocument();
+    ChspsDomDepthIterator* iter = ChspsDomDepthIterator::NewL( *dom.RootNode() );
+    CleanupStack::PushL( iter );
+
+    // Find a plugin node with the provided id attribute
+    ChspsDomNode* node = iter->First();
+    ChspsDomNode* prevNode = NULL;
+    TBool jobDone = EFalse;
+    while( node && !jobDone && prevNode != node )
+        {               
+        const TDesC8& name = node->Name();
+        
+        // Plugin element was found 
+        if ( name == KPluginElement )
+            {           
+            ChspsDomList& attrList = node->AttributeList();                 
+            ChspsDomAttribute* attr = static_cast<ChspsDomAttribute*>( attrList.FindByName(KPluginAttrUid) );           
+            if ( !attr )
+                {
+                // Mandatory information is missing for some reason (should be set at installation handler)!
+                // Exit with NULL
+                jobDone = ETrue;                
+                }
+            else
+                {
+                // Convert from (hex?) string into TUid presentation
+                const TUid uid = ConvertDescIntoUid( attr->Value() );                          
+                if ( aPluginUid == uid.iUid )
+                    {
+                    aInstanceCount++;
+                    }
+                }
+            }
+        
+        prevNode = node;        
+        node = iter->NextL();       
+        }
+    CleanupStack::PopAndDestroy( iter );    
+    }
+// -----------------------------------------------------------------------------
+// hspsServerUtil::ConvertDescIntoUid()
+//----------------------------------------------------------------------------
+//
+TUid hspsServerUtil::ConvertDescIntoUid(
+        const TDesC8& aStr )
+    {
+    TLex8 lex(aStr);
+    TUint pluginUid;
+    lex.Mark();
+    
+    if(lex.Peek() == '0')
+        {
+        lex.Inc();
+        }
+    if(lex.Peek() == 'x')
+        {
+        lex.Inc();
+        }
+    lex.Mark();
+    while (lex.Peek().IsHexDigit())
+        {
+        lex.Inc();
+        }
+    TPtrC8 uidToken = lex.MarkedToken();
+    TLex8 uidLex(uidToken);
+    TInt err = uidLex.Val(pluginUid,EHex);
+    
+    return TUid::Uid(pluginUid);
+    }
+//----------------------------------------------------------------------------
+// hspsServerUtil::FindChildNodeByTagL()
+// ----------------------------------------------------------------------------
+//
+ChspsDomNode* hspsServerUtil::FindChildNodeByTagL(
+    const TDesC8& aNodeTag, 
+    ChspsDomNode& aParentNode,
+    TInt& aIndex )
+    {
+    ChspsDomNode* node( NULL );
+    ChspsDomList& items = aParentNode.ChildNodes();
+    TInt length = items.Length();
+    node = NULL;
+    for ( TInt i = aIndex; i < length && node == NULL; i++ )
+        {
+        node = static_cast<ChspsDomNode*>( items.Item( i ) );
+        const TDesC8& name = node->Name();
+        if( name.Compare( aNodeTag ) != 0 )
+            {
+            node = NULL;
+            }
+        else
+            {
+            aIndex = i;
+            }
+        }
+    
+    return node;
+    }
+
+//----------------------------------------------------------------------------
+// hspsServerUtil::FindFilesL()
+// ----------------------------------------------------------------------------
+//
+void hspsServerUtil::FindFilesL(
+    const TDesC& aDirName, 
+    const TDesC& aFileName,
+    RArray <TFileName>& aFiles )
+    {
+    RFs fs;
+    CleanupClosePushL( fs );
+    User::LeaveIfError( fs.Connect() );
+    
+    // Find files from root directory
+    hspsServerUtil::FindFilesFromDirL( 
+            aDirName, 
+            aFileName, 
+            aFiles );
+    
+    // Directory scanner to browse directory structure
+    CDirScan* dirScan = CDirScan::NewL( fs );
+    CleanupStack::PushL( dirScan );
+    dirScan->SetScanDataL( 
+        aDirName, 
+        ( KEntryAttDir | KEntryAttMatchExclusive ), 
+        ESortNone );
+    
+    // Directory path where installation files are searched
+    TFileName dirName;
+
+    // Get first directory list
+    CDir* dirList( NULL );
+    dirScan->NextL( dirList );
+    // Find files from root directories
+    while ( dirList )
+        {
+        CleanupStack::PushL( dirList );
+        for ( TInt i = 0; i < dirList->Count(); i++ )
+            {
+            // Get directory path
+            dirName = dirScan->FullPath();
+            // Append directory entry
+            const TEntry& dirEntry = ( *dirList )[ i ];
+            dirName.Append( dirEntry.iName );
+            dirName.Append( KDoubleBackSlash );
+            hspsServerUtil::FindFilesFromDirL( 
+                dirName, 
+                aFileName, 
+                aFiles );
+            }
+        // Get next directory list
+        CleanupStack::PopAndDestroy( dirList );
+        dirScan->NextL( dirList );
+        }
+    
+    CleanupStack::PopAndDestroy( dirScan );
+
+    CleanupStack::PopAndDestroy(); // fs
+    
+    }
+
+//----------------------------------------------------------------------------
+// hspsServerUtil::FindFilesFromDirL()
+// ----------------------------------------------------------------------------
+//
+void hspsServerUtil::FindFilesFromDirL(
+    const TDesC& aDirName, 
+    const TDesC& aFileName,
+    RArray <TFileName>& aFiles )
+    {
+    RFs fs;
+    CleanupClosePushL( fs );
+    User::LeaveIfError( fs.Connect() );
+    
+    // File finder to search files from a directory
+    TFindFile fileFinder( fs );
+    
+    // Define drives where files are searched
+    TInt driveNumber;
+    TParse dirParser;
+    dirParser.Set( aDirName, NULL, NULL );
+    // Default drives C: and Z:
+    TInt findMask( 
+            KDriveAttExclude |
+            KDriveAttRemovable |
+            KDriveAttRemote );
+    if( RFs::CharToDrive( dirParser.Drive()[0], driveNumber ) == KErrNone )
+        {
+        if ( driveNumber == EDriveC )
+            {
+            // Search from C: drive
+            findMask = ( 
+                KDriveAttExclude | 
+                KDriveAttRemovable | 
+                KDriveAttRemote | 
+                KDriveAttRom );
+            }
+        else if ( driveNumber == EDriveZ )
+            {
+            // Search from Z: drive
+            findMask = KDriveAttRom;
+            }
+        }            
+    fileFinder.SetFindMask( findMask ); 
+    
+    // Find files from the directory entry
+    CDir* fileList( NULL );
+    fileFinder.FindWildByDir( aFileName, aDirName, fileList );
+    CleanupStack::PushL( fileList );
+    for ( TInt k = 0; fileList && k < fileList->Count(); k++ )
+        {
+        // Add found file with full path to file array
+        const TEntry& fileEntry = (*fileList)[k];
+        TFileName file;
+        file.Append( aDirName );
+        file.Append( fileEntry.iName );
+        aFiles.Append( file );
+        }
+    CleanupStack::PopAndDestroy( fileList );
+
+    CleanupStack::PopAndDestroy(); // fs
+    }
+
+// -----------------------------------------------------------------------------
+// hspsServerUtil::SetAttributeValueL
+// -----------------------------------------------------------------------------
+//
+void hspsServerUtil::SetAttributeValueL( 
+    const ChspsODT& aOdt,
+    const TDesC8& aNodeName,
+    const TDesC8& aAttrName, 
+    const TDesC8& aAttrValue,
+    const TDesC8& aSetAttrName,
+    const TDesC8& aSetAttrValue )
+    {
+    
+    ChspsDomDocument& dom = aOdt.DomDocument();
+    ChspsDomDepthIterator* iter = ChspsDomDepthIterator::NewL( *dom.RootNode() );   
+    CleanupStack::PushL( iter );
+                        
+    TBool nodeFound( EFalse );
+    ChspsDomNode* prevNode( NULL );
+    ChspsDomNode* node = iter->First();
+    while( node && 
+           prevNode != node )
+        {           
+        const TDesC8& name = node->Name();        
+        if ( name.CompareF( aNodeName ) == 0 )
+            {
+            // Node name match
+            ChspsDomList& attrList = node->AttributeList();
+            ChspsDomAttribute* attr = static_cast<ChspsDomAttribute*>( 
+                attrList.FindByName( aAttrName ) );               
+            if ( attr && attr->Value().CompareF( aAttrValue ) == 0 )
+                {
+                // Attribute name and value match - Defined node found
+                attr = static_cast<ChspsDomAttribute*>( 
+                    attrList.FindByName( aSetAttrName ) );
+                if ( attr )
+                    {
+                    // Updated attribute found - Update attribute value
+                    attr->SetValueL( aSetAttrValue );
+                    }
+                else
+                    {
+                    // Add new attribute
+                    AddAttributeDescL( *node, aSetAttrName, aSetAttrValue );
+                    }
+                nodeFound = ETrue;
+                }
+            }       
+        // Get next node
+        prevNode = node;    
+        node = iter->NextL();
+        }
+
+    if ( !nodeFound )
+        {
+        User::Leave( KErrNotFound );
+        }
+    
+    CleanupStack::PopAndDestroy( iter );
+        
+    }
+
+// -----------------------------------------------------------------------------
+// hspsServerUtil::GetAttributeValueL
+// -----------------------------------------------------------------------------
+//
+void hspsServerUtil::GetAttributeValueL( 
+    const ChspsODT& aOdt,
+    const TDesC8& aNodeName,
+    const TDesC8& aAttrName, 
+    const TDesC8& aAttrValue,
+    const TDesC8& aGetAttrName,
+    TPtrC8& aGetAttrValue )
+    {
+    __UHEAP_MARK;
+    
+    ChspsDomDocument& dom = aOdt.DomDocument();
+    ChspsDomDepthIterator* iter = ChspsDomDepthIterator::NewL( *dom.RootNode() );   
+    CleanupStack::PushL( iter );
+                        
+    TBool nodeFound( EFalse );
+    ChspsDomNode* prevNode( NULL );
+    ChspsDomNode* node = iter->First();
+    while( node && 
+           prevNode != node &&
+           !nodeFound )
+        {           
+        const TDesC8& name = node->Name();        
+        if ( name.CompareF( aNodeName ) == 0 )
+            {
+            // Node name match
+            ChspsDomList& attrList = node->AttributeList();
+            ChspsDomAttribute* attr = static_cast<ChspsDomAttribute*>( 
+                attrList.FindByName( aAttrName ) );               
+            if ( attr && attr->Value().CompareF( aAttrValue ) == 0 )
+                {
+                // Attribute name and value match - Defined node found
+                attr = static_cast<ChspsDomAttribute*>( 
+                    attrList.FindByName( aGetAttrName ) );
+                if ( attr )
+                    {
+                    // Updated 
+                    aGetAttrValue.Set( attr->Value() );
+                    nodeFound = ETrue;
+                    }
+                }
+            }       
+        // Get next node
+        prevNode = node;    
+        node = iter->NextL();
+        }
+
+    if ( !nodeFound )
+        {
+        User::Leave( KErrNotFound );
+        }
+    
+    CleanupStack::PopAndDestroy( iter );
+    
+    __UHEAP_MARKEND;
+    
+    }
+
+// -----------------------------------------------------------------------------
+// hspsServerUtil::CheckResourceFilesL
+// -----------------------------------------------------------------------------
+//
+void hspsServerUtil::CheckResourceFilesL( 
+    const ChspsODT& aOdt,
+    const TInt aConfUid )
+    {
+    __UHEAP_MARK;
+
+    // Convert configuration UID to decimal string
+    TBuf<10> confUid; 
+    _LIT( KFormat, "%D" );
+    confUid.AppendFormat( KFormat, aConfUid );
+
+    RFs fs;
+    CleanupClosePushL( fs );
+    User::LeaveIfError( fs.Connect() );
+    
+    TInt resCount = aOdt.ResourceCount();
+    for ( TInt i = 0; i < resCount; i++ )
+        {
+        // Check if resource file belongs to defined configuration
+        // (file path contains configuration UID string)
+        ChspsResource& res = aOdt.ResourceL( i );
+        TPtrC resFile = res.FileName();
+        if ( resFile.FindC( confUid ) != KErrNotFound )
+            {
+            // Check that resource files exists
+            if ( !BaflUtils::FileExists( fs, resFile ) )
+                {
+                User::Leave( KErrNotFound );
+                }
+            }
+        }
+    
+    CleanupStack::PopAndDestroy(); // fs
+    
+    __UHEAP_MARKEND;
+    
+    }
+
+// -----------------------------------------------------------------------------
+// hspsServerUtil::GetConfigurationVersionL
+// -----------------------------------------------------------------------------
+//
+void hspsServerUtil::CheckConfigurationVersionL( 
+    ChspsODT& aOdt,
+    const TInt aConfUid,
+    const TDesC& aVersion )
+    {
+    __UHEAP_MARK;
+
+    // Create configuration UID string
+    TBuf8<10> confUid;    
+    _LIT8( KFormat8, "%X" );
+    _LIT8( KHexPrefix, "0x" );
+    confUid.Append( KHexPrefix );
+    confUid.AppendFormat( KFormat8, aConfUid );
+
+    // Find configuration node
+    ChspsDomNode* confNode = hspsServerUtil::FindNodeByAttributeL(
+        aOdt,
+        KConfigurationElement,
+        KConfigurationAttrUid,
+        confUid );
+    
+    if ( confNode )
+        {
+        ChspsDomList& attrList = confNode->AttributeList();
+        ChspsDomAttribute* attr = static_cast<ChspsDomAttribute*>( 
+            attrList.FindByName( KConfigurationAttrVersion ) );               
+        if ( attr )
+            {
+            HBufC8* tv = HBufC8::NewLC( aVersion.Length() );
+            TPtr8 tvPtr( tv->Des() );
+            tvPtr.Copy( aVersion );
+            // Check configuration version
+            if ( attr->Value().Compare( tvPtr ) != 0 )
+                {
+                // Configuration not supported
+                User::Leave( KErrNotSupported );
+                }
+            CleanupStack::PopAndDestroy( tv );
+            }
+        else
+            {
+            // Invalid configuration
+            User::Leave( KErrGeneral );
+            }
+        }
+    else
+        {
+        // Configuration not found
+        User::Leave( KErrNotFound );
+        }
+    
+    __UHEAP_MARKEND;
+    }
+
+// -----------------------------------------------------------------------------
+// hspsServerUtil::EditPluginNodeActivityL
+// -----------------------------------------------------------------------------
+//
+void hspsServerUtil::EditPluginNodeActivityL( ChspsDomNode* aRootNode,
+                                              const TNodeEditMode aEditMode,
+                                              TInt aDepth  )
+    {
+    // This wrapper exists to prevent modification of original aDepth
+    // (_EditPluginNodeActivityL uses TInt reference for depth count and
+    // modifies it).
+    hspsServerUtil::_EditPluginNodeActivityL( aRootNode, aEditMode, aDepth );
+    }
+
+// -----------------------------------------------------------------------------
+// hspsServerUtil::_EditPluginNodeActivityL
+// -----------------------------------------------------------------------------
+//
+void hspsServerUtil::_EditPluginNodeActivityL( ChspsDomNode* aRootNode,      
+                                               const TNodeEditMode aEditMode,
+                                               TInt& aDepth )
+    {
+    // Sanity check.
+    if( !aRootNode )
+        {
+        return;
+        }
+
+    // Recursion depth logic.
+    if( aDepth == RECURSION_DEPTH_INFINITE )
+        {
+        // Work as normal. Recurse as far as object tree exists.
+        }
+    else if( aDepth > 0 )
+        {
+        // Depth was defined. Decrement by one.
+        aDepth--;
+        }
+    else if( aDepth == 0 )
+        {
+        // No more recursion.
+        return;
+        }
+    
+    // Iterate childs    
+    ChspsDomList& childs = aRootNode->ChildNodes();    
+    for( int i = 0; i < childs.Length(); i++ )
+        {
+        ChspsDomNode* child = static_cast<ChspsDomNode*>( childs.Item( i ) );         
+        
+        // Modify node activity based on selected edit mode.
+        if( child->Name().CompareF( KPluginElement ) == 0 )
+            {
+            if( aEditMode == EActivateFirst && i == 0 )
+                {
+                hspsServerUtil::AddAttributeDescL( *child,
+                                                   KPluginAttrActive,
+                                                   KPluginActiveStateActive );                
+                }
+            else 
+                {
+                hspsServerUtil::AddAttributeDescL( *child,
+                                                   KPluginAttrActive,
+                                                   KPluginActiveStateNotActive );                
+                }
+            }
+
+        // Recurse.
+        _EditPluginNodeActivityL( child,
+                                  aEditMode,
+                                  aDepth );
+        }    
+    }
+
+// -----------------------------------------------------------------------------
+// hspsServerUtil::GetActivePluginNode
+// -----------------------------------------------------------------------------
+//
+ChspsDomNode* hspsServerUtil::GetActivePluginNode( ChspsDomNode* aParentNode )
+    {
+    // Sanity check.
+    if( !aParentNode )
+        {
+        return NULL;
+        }
+
+    // Return structure.
+    ChspsDomNode* activeNode = NULL;
+    
+    // Iterate childs    
+    ChspsDomList& childs = aParentNode->ChildNodes();    
+    for( int i = 0; i < childs.Length(); i++ )
+        {
+        ChspsDomNode* child = static_cast<ChspsDomNode*>( childs.Item( i ) );         
+        
+        // Only for plugin elements.
+        if( child->Name().CompareF( KPluginElement ) == 0 )
+            {
+            // 1. Check if has activity attribute.            
+            ChspsDomList& attrList = child->AttributeList();            
+            ChspsDomAttribute* pluginActivityAttr = 
+                static_cast<ChspsDomAttribute*>( attrList.FindByName( KPluginAttrActive ) );                
+            if( pluginActivityAttr )
+                {
+                // 2. Check whether node is active.
+                if( pluginActivityAttr->Value().CompareF( KPluginActiveStateActive ) == 0 )
+                    {
+                    // 3. Active node found. assign and break.
+                    activeNode = child;
+                    break;
+                    }
+                }                
+            }
+        }
+    
+    return activeNode;
+    }
+
+// -----------------------------------------------------------------------------
+// hspsServerUtil::GetPluginId
+// -----------------------------------------------------------------------------
+//
+TInt hspsServerUtil::GetPluginId( ChspsDomNode* aNode )
+    {
+    // Sanity check.
+    if( !aNode )
+        {
+        return KErrArgument;
+        }
+    
+    // Return value.
+    TInt pluginId = KErrNotFound;
+
+    // Find out plugin id.
+    ChspsDomList& attrList = aNode->AttributeList();                                                                                                      
+    ChspsDomAttribute* pluginIdAttr = 
+        static_cast<ChspsDomAttribute*> ( attrList.FindByName( KPluginAttrId ) );        
+    if( pluginIdAttr )
+        {
+        // Found.
+        const TDesC8& pluginIdValue = pluginIdAttr->Value();
+        pluginId = DecString2Int( pluginIdValue );  
+        }
+    
+    return pluginId;
+    }
+
+// -----------------------------------------------------------------------------
+// hspsServerUtil::GetPluginUid
+// -----------------------------------------------------------------------------
+//
+TUid hspsServerUtil::GetPluginUid( ChspsDomNode* aNode )
+    {
+    // Sanity check.
+    if( !aNode )
+        {
+        return KNullUid;
+        }
+    
+    // Read Uid from attribute list.
+    ChspsDomList& attrList = aNode->AttributeList();
+    ChspsDomAttribute* pluginUidAttr = 
+        static_cast<ChspsDomAttribute*> ( attrList.FindByName( KPluginAttrUid ) );                                                            
+
+    // Sanity check.
+    if( !pluginUidAttr )
+        {
+        return KNullUid;
+        }
+    
+    // Convert uids from string to numeric format
+    const TDesC8& pluginUidValue = pluginUidAttr->Value();                    
+    const TUid pluginUid = hspsServerUtil::ConvertDescIntoUid(pluginUidValue);    
+    
+    // Return result.
+    return pluginUid;
+    }
+
+// -----------------------------------------------------------------------------
+// hspsServerUtil::GetPluginIdsByUidL
+// -----------------------------------------------------------------------------
+//
+void hspsServerUtil::GetPluginIdsByUidL(
+        const ChspsODT& aAppODT,        
+        const TUid aPluginUid,
+        RArray<TInt>& aPluginIds )       
+    {       
+    ChspsDomDocument& dom = aAppODT.DomDocument();
+    ChspsDomDepthIterator* iter = ChspsDomDepthIterator::NewL( *dom.RootNode() );
+    CleanupStack::PushL( iter );
+
+    // Find a plugin node with the provided id attribute
+    ChspsDomNode* node = iter->First();
+    ChspsDomNode* prevNode = NULL;
+    TBool jobDone = EFalse;
+    while( node && !jobDone && prevNode != node )
+        {               
+        const TDesC8& name = node->Name();
+        
+        // Plugin element was found 
+        if ( name == KPluginElement )
+            {           
+            ChspsDomList& attrList = node->AttributeList();                 
+            ChspsDomAttribute* attr = static_cast<ChspsDomAttribute*>( attrList.FindByName(KPluginAttrUid) );           
+            if ( !attr )
+                {
+                // Mandatory information is missing for some reason (should be set at installation handler)!
+                // Exit with NULL
+                jobDone = ETrue;                
+                }
+            else
+                {
+                // Convert from (hex?) string into TUid presentation
+                const TUid uid = ConvertDescIntoUid( attr->Value() );                          
+                if ( aPluginUid == uid )
+                    {
+                    const TDesC8& strPluginId = node->AttributeValue(KPluginAttrId);
+                    TInt pluginId = DecString2Int( strPluginId );           
+                    aPluginIds.AppendL( pluginId );                    
+                    }
+                }
+            }
+        
+        prevNode = node;        
+        node = iter->NextL();       
+        }
+    
+    CleanupStack::PopAndDestroy( iter );    
+    }
+
+// -----------------------------------------------------------------------------
+// hspsServerUtil::GetParentNode
+// -----------------------------------------------------------------------------
+//
+ChspsDomNode* hspsServerUtil::GetParentNode(
+    const ChspsDomNode& aNode,
+    const TDesC8& aNodeName,
+    const TDesC8& aAttrName,
+    const TDesC8& aAttrValue )       
+    {
+    ChspsDomNode* parent = aNode.Parent();
+    TBool found( EFalse );
+    while ( parent && !found )
+        {
+        if ( parent->Name().CompareF( aNodeName ) == 0 )
+            {
+            ChspsDomList& attrList = parent->AttributeList();
+            ChspsDomAttribute* attr = static_cast<ChspsDomAttribute*>( attrList.FindByName( aAttrName ) );
+            if ( attr && attr->Value().CompareF( aAttrValue ) == 0 )
+                {
+                found = ETrue;
+                }
+            }
+        if ( !found )
+            {
+            // Get next level parent node
+            parent = parent->Parent();
+            }
+        }
+    
+    return parent;
+    }
+
+// -----------------------------------------------------------------------------
+// hspsServerUtil::IsLogoFile
+// -----------------------------------------------------------------------------
+//
+TBool hspsServerUtil::IsLogoFile(
+        const TDesC& aFileDeclaration,
+        TFileName& aFilename )        
+    {
+    _LIT(KDeclarationSkin, "SKIN(");        // prefixes for filelogo/filepreview values 
+    _LIT(KDeclarationMif, "MIF(");
+    _LIT(KDeclarationUid, "UID(");
+        
+    aFilename = KNullDesC();
+    if ( aFileDeclaration.FindF( KDeclarationSkin ) >= 0  
+            || aFileDeclaration.FindF( KDeclarationMif ) >= 0             
+            || aFileDeclaration.FindF( KDeclarationUid ) >= 0 )
+        {                
+        // pick filename from the mif declaration if it's available (e.g. "skin(<id> <id>):mif(<path> <id> <id>)")
+        TInt mifOffset = aFileDeclaration.FindF( KDeclarationMif );
+        if ( mifOffset >= 0 )            
+            {              
+            aFilename = aFileDeclaration.Mid( mifOffset + KDeclarationMif().Length() );
+            aFilename.TrimAll();                
+                        
+            // drop everything before the first bitmap index
+            TInt endPos = aFilename.Locate( ' ' );
+            if ( endPos > 1 )
+                {
+                aFilename = aFilename.Left( endPos );                                
+                }                                                      
+            }                                       
+        }
+    else
+        {
+        // store possible icon path
+        aFilename.Copy( aFileDeclaration );
+        }
+    
+    return ( aFilename.Length() > 0 );
+    }
+
+// -----------------------------------------------------------------------------
+// hspsServerUtil::GetResourcesForLanguageL
+// -----------------------------------------------------------------------------
+void hspsServerUtil::GetResourcesForLanguageL(
+        ChspsODT& aODT,        
+        const TInt aConfUid,
+        const TLanguage aActiveLanguage,
+        RPointerArray<ChspsResource>& aWidgetResources )
+    {    
+    const TInt resourceCount = aODT.ResourceCount();
+            
+    // pick resources with the active language       
+    for( TInt resourceIndex = 0; resourceIndex < resourceCount; resourceIndex++ )
+        {
+        ChspsResource& resource = aODT.ResourceL( resourceIndex );        
+        if ( resource.ConfigurationUid() == aConfUid &&
+             resource.Language() == aActiveLanguage &&
+             resource.FileName().FindF( KSourcesFolder ) > 0 )                
+            {                                
+            aWidgetResources.Append( &resource );
+            }
+        }        
+    
+    // Resources for language none or language test.    
+    for( TInt resourceIndex = 0; resourceIndex < resourceCount; resourceIndex++ )
+        {
+        ChspsResource& resource = aODT.ResourceL( resourceIndex );
+        if ( resource.ConfigurationUid() == aConfUid &&
+             resource.FileName().FindF( KSourcesFolder ) > 0 )
+            {
+            if( resource.Language() == ELangTest || resource.Language() == ELangNone )
+                {
+                // Checking also that not going to overwrite existing localized resource.                
+                TBool localizedVersionAvailable = EFalse;
+                for( TInt checkIndex = 0; checkIndex < aWidgetResources.Count(); checkIndex++ )
+                    {
+                    ChspsResource* checkResource = aWidgetResources[ checkIndex ];
+                    if( !checkResource )
+                        {
+                        continue;
+                        }
+                    
+                    TParsePtrC resource1FullPath( resource.FileName() );
+                    TParsePtrC resource2FullPath( checkResource->FileName() );
+                    
+                    if( resource1FullPath.NameAndExt().CompareF( resource2FullPath.NameAndExt() ) == 0 )                 
+                        {
+                        localizedVersionAvailable = ETrue;
+                        break;
+                        }
+                    }
+            
+                if( !localizedVersionAvailable )
+                    {
+                    aWidgetResources.Append( &resource );
+                    }
+                }
+            }
+        }           
+    }
+
+// -----------------------------------------------------------------------------
+// hspsServerUtil::EnoughDiskSpaceAvailable
+// -----------------------------------------------------------------------------
+TInt hspsServerUtil::EnoughDiskSpaceAvailableL(
+        ChspsODT& aODT,        
+        const TLanguage aActiveLanguage,
+        RFs& aFs,
+        const TDriveNumber aDriveNumber,
+        const TInt aAdditionalDiskSpace )
+    {        
+    __UHEAP_MARK;
+    
+    TInt err = KErrNone;   
+    
+    // Retrieve data for needed resource files.
+    RPointerArray<ChspsResource> widgetResources; // Objects are not owned.
+    CleanupClosePushL( widgetResources );
+    
+    GetResourcesForLanguageL( 
+        aODT,
+        aODT.ThemeUid(),
+        aActiveLanguage,
+        widgetResources );
+    
+    TInt requiredDiskSpace = aAdditionalDiskSpace;
+    
+    // Calculate disk space required for resources.
+    for( TInt i = 0; i < widgetResources.Count() && !err; i++ )
+        {      
+        ChspsResource* resource = widgetResources[i];
+        if( !resource )
+            {
+            continue;
+            }
+                         
+        TEntry entryData;
+        err = aFs.Entry( resource->FileName(), entryData );        
+        requiredDiskSpace += entryData.iSize;  
+        }
+    
+    widgetResources.Reset();
+    CleanupStack::PopAndDestroy( 1, &widgetResources );    
+    
+    // Check if calculated space is available.
+    if( !err &&
+        SysUtil::DiskSpaceBelowCriticalLevelL( &aFs,
+                requiredDiskSpace,
+                aDriveNumber ) )
+        {
+        err = KErrDiskFull;
+        }
+        
+    __UHEAP_MARKEND;
+    
+    return err;
+    }
+// -----------------------------------------------------------------------------
+// Finds a node from a dom document.
+// Looks for the next node tag.
+// -----------------------------------------------------------------------------
+ChspsDomNode* hspsServerUtil::FindNodeByTagL( 
+        const TDesC8& aNodeTag,
+        ChspsDomNode& aDomNode )
+    {
+    ChspsDomDepthIterator* iter = ChspsDomDepthIterator::NewL( aDomNode );
+    CleanupStack::PushL( iter );
+    ChspsDomNode* targetNode( NULL );
+    ChspsDomNode* node = iter->First();
+    TBool found = EFalse;
+    while( !found && node )
+        {
+        const TDesC8& name = node->Name();
+        if ( name.Compare( aNodeTag ) == 0 )
+            {  
+            found = ETrue;
+            targetNode = node;
+            }
+        node = iter->NextL();
+        }   
+    CleanupStack::PopAndDestroy( iter );
+    return targetNode;
+    } 
+
+
+// -----------------------------------------------------------------------------
+ // hspsServerUtil::FindFile
+ // Eclipsing support for customization 
+ // -----------------------------------------------------------------------------
+ //
+ TInt hspsServerUtil::FindFile(
+         RFs& aFs,
+         const TDesC& aPath,
+         const TDesC& aFilename,        
+         const TBool aFindFromUdaEmmcDrives,
+         TFileName& aDrivePathName )
+     {   
+     TInt err = KErrNotFound;
+     
+     TParsePtrC parser( aPath );
+     const TPath path = parser.Path();
+          
+     TFileName filename( aFilename );         
+     if( filename.Length() == 0 )
+         {
+         filename.Copy( parser.NameAndExt() );
+         }
+     
+     if( filename.Length() > 0 && path.Length() > 0 )
+         {              
+         // Find the input file, search from the user area (UDA) first, 
+         // exclude external/remote drives from the search - otherwise end-users  
+         // could introduce fixed configurations (e.g. operator locks wouldn't work)
+         TFindFile fileFinder( aFs );
+         fileFinder.SetFindMask( 
+             KDriveAttExclude|KDriveAttRemovable|KDriveAttRemote|KDriveAttSubsted );
+         if( aFindFromUdaEmmcDrives )
+             {
+             TInt drive = hspsServerUtil::GetEmmcDrivePath( aFs );
+             if ( drive != KErrNotFound )
+                 {
+                 aFs.SetSessionToPrivate( drive );
+                 }
+             }
+         else 
+             {
+             aFs.SetSessionToPrivate( EDriveZ );
+             }
+         err = fileFinder.FindByDir( filename, path );
+         aFs.SetSessionToPrivate( EDriveC );     
+         if( !err )          
+             {         
+             // Return the path with a drive reference 
+             aDrivePathName = fileFinder.File();        
+             TParsePtrC drvParser( aDrivePathName );
+             if( !drvParser.DrivePresent() )
+                 {             
+                 err = KErrNotFound;
+                 }
+             }
+         }
+              
+     return err;
+     }
+     
+// -----------------------------------------------------------------------------
+// hspsServerUtil::ResolveLogoPathL
+// -----------------------------------------------------------------------------
+void hspsServerUtil::PopulateLogoPathsL(
+        const TDesC& aLogoDeclaration,
+        const TUint aAppUid,
+        RBuf& aTargetPath,
+        RBuf& aSourcePath,
+        RBuf& aUpdatedDeclaration)
+    {        
+    // Process widget types only 
+    if ( aLogoDeclaration.Length() && aAppUid > 0 )
+        {                          
+        // Get possible file name from the optional logo declaration
+        // and if found, populate the paths and update the declaration 
+        TFileName filename;
+        if( IsLogoFile( aLogoDeclaration, filename ) )
+            {      
+            // Get client's private directory                
+            _LIT( KClientPrivatePath, "c:\\private\\%X\\");
+            TPath clientPath;            
+            clientPath.Format( KClientPrivatePath, aAppUid );
+                                    
+            // Updated logo declaration
+            TInt offset = aLogoDeclaration.FindF( filename );                       
+            __ASSERT_DEBUG( offset != KErrNotFound, User::Leave( KErrCorrupt ) );            
+            if( aLogoDeclaration.Length() + aLogoDeclaration.Mid( offset ).Length() < KMaxFileName )
+                {
+                aUpdatedDeclaration.Copy( aLogoDeclaration );
+                aUpdatedDeclaration.Insert( offset, clientPath );
+                                                               
+                // Set path and name of the target file            
+                if( clientPath.Length() + filename.Length() < KMaxFileName )
+                    {
+                    aTargetPath.Copy( clientPath );
+                    aTargetPath.Append( filename );
+                    
+                    // Set name of the source file
+                    _LIT( KServerPrivateFolder, "c:\\private\\200159c0\\themes\\" );
+                    if( KServerPrivateFolder().Length() + filename.Length() < KMaxFileName )
+                        {                       
+                        aSourcePath.Copy( KServerPrivateFolder );
+                        aSourcePath.Append( filename );                        
+                        }
+                    }
+                }
+                        
+            }
+        }
+    }     
+
+// -----------------------------------------------------------------------------
+// hspsServerUtil::FindResourcesL
+// -----------------------------------------------------------------------------
+void hspsServerUtil::FindResourcesL(
+        RFs& aFs,
+        const RArray<TInt>& aDriveArray, 
+        const TDesC& aPath,        
+        RPointerArray<HBufC>& aFileArray,
+        CArrayFixFlat<TInt>* aDeviceLanguages,
+        TBool aRecursive )
+    {
+    // Scan internal drives only 
+    TFindFile fileFinder( aFs );    
+    fileFinder.SetFindMask( KDriveAttExclude|KDriveAttRemovable|KDriveAttRemote|KDriveAttSubsted );
+         
+    TParsePtrC parser( aPath );
+    
+    // Loop the provided disk drives
+    for( TInt driveIndex=0; driveIndex < aDriveArray.Count(); driveIndex++ )
+        {
+        TChar driveChar;
+        User::LeaveIfError( RFs::DriveToChar( aDriveArray[driveIndex], driveChar ) );
+        TBuf16<2> driveBuf(2);
+        driveBuf[0] = TUint( driveChar );
+        driveBuf[1] = TUint( TChar(':') );
+                        
+        TPath path;        
+        path.Copy( driveBuf );        
+        path.Append( parser.Path() );
+                       
+        // Find files from the drive and path
+        CDir* dirList( NULL );        
+        fileFinder.FindWildByPath( path, NULL, dirList );
+        if ( dirList )
+            {
+            CleanupStack::PushL( dirList );
+                       
+            const TInt count = dirList->Count();          
+            for( TInt entryIndex = 0; entryIndex < count; entryIndex++ )
+                {
+                const TEntry& entry = (*dirList)[ entryIndex ];                        
+                                                         
+                TFileName file( path );              
+                file.Append( entry.iName );
+                
+                if( entry.IsDir() )
+                    {                               
+                    if( aDeviceLanguages )
+                        {
+                        TInt dirLanguage = 0;
+                        TLex lex( entry.iName );
+                        TBool skipDir = ETrue;
+                        if( lex.Val( dirLanguage ) == KErrNone && dirLanguage >= ELangTest )
+                            {                   
+                            for( TInt i=0; i < aDeviceLanguages->Count(); i++ )
+                                {
+                                TInt supportedLanguage = aDeviceLanguages->At( i );
+                                if( supportedLanguage == dirLanguage )
+                                    {
+                                    skipDir = EFalse;
+                                    break;
+                                    }
+                                }
+                            }
+                        if( skipDir )
+                            {
+                            continue;
+                            }
+                        }
+              
+                  file.Append( KDoubleBackSlash );
+                  }
+                                
+              if( entry.IsDir() && aRecursive )
+                  {                   
+                  // Find files from the directory and drive
+                  RArray<TInt> driveArray;
+                  CleanupClosePushL( driveArray );                  
+                  driveArray.Append( aDriveArray[driveIndex] );                                   
+                  FindResourcesL( aFs, driveArray, file, aFileArray, NULL );   
+                  CleanupStack::PopAndDestroy( &driveArray );
+                  }
+              else
+                  {                                                  
+                  HBufC* nameBuf = file.AllocLC();                
+                  aFileArray.AppendL( nameBuf );
+                  CleanupStack::Pop( nameBuf );                      
+                  }              
+              } 
+          
+          CleanupStack::PopAndDestroy( dirList );
+          dirList = 0;
+          } // dirlist
+    
+        } // driveIndex    
+    }
+
+// -----------------------------------------------------------------------------
+// hspsServerUtil::GetInstalledLanguagesL
+// -----------------------------------------------------------------------------
+void hspsServerUtil::GetInstalledLanguagesL(
+        CArrayFixFlat<TInt>*& aLanguages )
+    {
+    User::LeaveIfError( SysLangUtil::GetInstalledLanguages( aLanguages ) );
+    CleanupStack::PushL( aLanguages );
+    
+    const TInt testLang = (TInt)ELangTest;
+    
+    TBool isIncluded = EFalse;           
+    for( TInt i = 0; i < aLanguages->Count(); i++ )
+        {
+        if( aLanguages->At( i ) == testLang )
+            {    
+            isIncluded = ETrue;            
+            break;
+            }
+        }    
+    
+    if( !isIncluded )
+        {
+        aLanguages->InsertL( 0, testLang );        
+        }
+    
+    CleanupStack::Pop( aLanguages );
+    }
+    
+
+// -----------------------------------------------------------------------------
+// hspsServerUtil::GetEmmcDrivePathL
+// -----------------------------------------------------------------------------
+//
+TInt hspsServerUtil::GetEmmcDrivePath( RFs& aFs )
+    {
+    TInt drive = KErrNotFound;
+    if ( DriveInfo::GetDefaultDrive( 
+            DriveInfo::EDefaultMassStorage, drive ) == KErrNone )
+        {
+        TUint status;
+        if ( DriveInfo::GetDriveStatus( aFs, drive, status ) == KErrNone )
+            {
+            if ( status & DriveInfo::EDriveInternal )
+                {
+                return drive;
+                }
+            }
+        }
+    return KErrNotFound;
+    }          
+
+// -----------------------------------------------------------------------------
+// Removes plugin resources from the provided ODT
+// -----------------------------------------------------------------------------
+//
+TInt hspsServerUtil::RemovePluginResourcesL(
+        ChspsODT& aAppODT,
+        const TInt aPluginUid )        
+    {            
+    // Loop resources of the application configuration            
+    for(TInt aresIndex = 0; aresIndex < aAppODT.ResourceCount(); aresIndex++ )
+        {
+        ChspsResource& ares = aAppODT.ResourceL( aresIndex );
+                                
+        // If the plugin resource was found at  resource list of the application configuration                        
+        if ( ares.ConfigurationUid() == aPluginUid )
+            {
+            // Deletes resource from the application configuration
+            aAppODT.DeleteResourceL( aresIndex );
+            aresIndex--;
+            }
+        
+        }                
+    
+    return KErrNone;
+    }
+
+// -----------------------------------------------------------------------------
+// hspsServerUtil::hspsServerUtil
+// -----------------------------------------------------------------------------
+//
+hspsServerUtil::hspsServerUtil() 
+    {
+    // Empty. Should never be called.
+    }            
+    
+// end of file