kernel/eka/drivers/hcr/hcr_pil.cpp
author Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
Thu, 07 Jan 2010 13:38:45 +0200
changeset 6 0173bcd7697c
parent 4 56f325a607ea
child 14 5d2844f35677
permissions -rw-r--r--
Revision: 201001 Kit: 201001

// Copyright (c) 2008-2009 Nokia Corporation and/or its subsidiary(-ies).
// All rights reserved.
// This component and the accompanying materials are made available
// under the terms of the License "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:
// Hardware Configuration Respoitory Platform Independent Layer (PIL)
//


// -- INCLUDES ----------------------------------------------------------------


#include "hcr_debug.h"

#include <e32def.h>
#include <e32err.h>
#include <e32des8.h>
#include <e32cmn.h>

#include <nkern/nkern.h>
#include <kernel/kernel.h>

#include <e32rom.h>
#include <plat_priv.h>

#include <kernel/kernboot.h>


#include "hcr_hai.h"
#include "hcr_pil.h"

// -- GLOBALS -----------------------------------------------------------------

GLDEF_C HCR::HCRInternal gHCR;

#ifdef HCR_TRACE
GLDEF_C TBuf<81> gTraceBuffer;
#endif


// -- Local functions prototypes
/**
Retrive Repository file address stored in the iHcrFileAddress field of ROM Image header.
If this filed is zero or it is equal with a special value then it keeps the original vaule of 
aRepos parameter and signals it with the retun value.


@param aRepos     		The reference to a repository variable    
@return	KErrNone 		if successful, the aRepos parameter references to the file in ROM Image.
        KErrNotFound 	if the ROM Image header contains zero or a special value as the repository file address


*/    
LOCAL_C TInt LocateCoreImgRepository(HCR::TRepository*& aRepos);

/**
This method transfer the value of aFileName to ROM Image conform file name string. 
Retrive the variant dependent ROM Root directory address.
Search the file in \sys\bin directory and if it doesn't exists there it try to find it in \sys\Data.


@param 	aRepos     			The reference to a repository variable.
				aFileName			  The name of the new repository file without path. '\0' terminated c-style string.
    
@return	KErrNone 			if successful
        KErrNotFound 		if file not found in \sys\bin or \sys\Data


*/    
LOCAL_C TInt SearchCoreImgRepository(HCR::TRepository* & aRepos, const TText * aFileName);

/**
Scanning a given directory for the given entry name. The entry name can be sub-directory or file.

@param 	aActDir     		Pointer to curretn directory in the ROM Image directory tree
		aFileName			File to be search
		aEntry				If the file found this referenced to proper directory entry
    
@return	 KErrNone		 			if the entry found
         KErrNotFound					if the entry not found
*/    

LOCAL_C TInt SearchEntryInTRomDir(const TRomDir* aActDir, const TPtrC aFileName, TRomEntry* &aEntry);


// -- WINS Specific ----------------------------------------------------------

#ifdef __WINS__

// Set to ensure Rom Hdr dependency does not break compilation in 
// LocateCoreImgRepository() at the end of this file.
// Undef incase it is set in MMP file, avoids compiler warning.
//
#undef HCRTEST_COREIMG_DONTUSE_ROMHDR
#define HCRTEST_COREIMG_DONTUSE_ROMHDR

#endif

// -- FUNCTIONS ---------------------------------------------------------------

/**
 Returns 1 when a1 > a2
 Returns -1 when a1 < a2
 Returns 0 when identical.
 */
TInt CompareSSettingIds(const HCR::TSettingId& a1, const HCR::SSettingId& a2)    
	{
    // HCR_FUNC("CompareSSettingIds");
    if (a1.iCat > a2.iCat)
        return (1); // HCR_TRACE_RETURN(1);
    if (a1.iCat < a2.iCat)
        return (-1); // HCR_TRACE_RETURN(-1);
    
    // Categories are the same at this point, check keys.
    if (a1.iKey > a2.iKey)
        return (1); // HCR_TRACE_RETURN(1);
    if (a1.iKey < a2.iKey)
        return (-1); // HCR_TRACE_RETURN(-1);
   
    // Both Categories and jeys are the same here.
    return (0); // HCR_TRACE_RETURN(0);
    }

#ifdef __EPOC32__
TBool ROMAddressIsInUnpagedSection(const TLinAddr address)
	{
    HCR_FUNC("ROMAddressIsInUnpagedSection");
	
	const TRomHeader& romHdr = Epoc::RomHeader();
	TLinAddr romBase = romHdr.iRomBase;

	HCR_TRACE1("--- address to check if in unpaged ROM section = 0x%8x", address);
	HCR_TRACE2("--- iRomSize (0x%8x), iPageableRomStart (0x%8x), ", romHdr.iRomSize, romHdr.iPageableRomStart);

	if ((address < romBase) || (romBase > romBase+romHdr.iRomSize))
		return EFalse;
	if (romHdr.iPageableRomStart == 0)
		return ETrue;
	if (address < romBase+romHdr.iPageableRomStart)
		return ETrue;
	return EFalse;
	}
#endif


TInt CompareByCategory(const HCR::TCategoryUid aCatId, const HCR::SSettingId& aSetId)    
    {
    //HCR_FUNC("CompareByCategory");
    if (aCatId > aSetId.iCat)
        return (1); // HCR_TRACE_RETURN(1);
    if (aCatId < aSetId.iCat)
        return (-1); // HCR_TRACE_RETURN(-1);
    
    // Both Categories and jeys are the same here.
    return (0); 
    }

/*
 * SafeArray TSa class object destructor. It delets the allocated in the heap
 * memory and set the instance pointer to NULL. See also TSa class definition
 * in hcr_pil.h.
 */
template<typename T>
    HCR::TSa<T>::~TSa()
    {
    delete[] iSa;
    iSa = NULL;
    }

/**
 * operator=() changes the memory ownership by   
 * reinitiazing SafeArray class object with the address to   
 * already allocated array.
 */
template<typename T>
   HCR::TSa<T>& HCR::TSa<T>::operator=(T* aP)
    {
    delete[] iSa;
    iSa = aP; 
    return (*this);
    }


// -- METHODS -----------------------------------------------------------------
//
// HCRInternal

HCR::HCRInternal::HCRInternal()
   : iStatus(EStatConstructed), iVariant(0), iVariantStore(0), iCoreImgStore(0), iOverrideStore(0)
    {
    HCR_FUNC("HCRInternal(Defualt)");
    }

HCR::HCRInternal::HCRInternal(HCR::MVariant* aVar)
   : iVariant(aVar), iVariantStore(0), iCoreImgStore(0), iOverrideStore(0)
    {
    HCR_FUNC("HCRInternal");
    }
    
HCR::HCRInternal::~HCRInternal()
    {
    HCR_FUNC("~HCRInternal");
    }
   
TUint32 HCR::HCRInternal::GetStatus()
    {
    HCR_FUNC("GetStatus");
    return iStatus;
    }
  
    
TInt HCR::HCRInternal::Initialise()
    {
    HCR_FUNC("HCRInternal::Initialise");
    
    TAny* store = 0; 
    TInt err = 0;
	
	// Variant PSL object must exist before PIL initalised.
	if (iVariant == 0) {
 			err = KErrGeneral; goto failed; }

	// Inform the PSL that we are initialising, give them an opportunity to do
	// initialisation work too.
    err = iVariant->Initialise(); 
    if (err != KErrNone)
    	goto failed;
   
    iStatus = EStatVariantInitialised;
    
    // Ask the PSL for the address of the SRepositoryCompiled object. PSL 
    // can return KErrNotSupported & NULL if compiled repository not 
	// used/support by PSL.
    err = iVariant->GetCompiledRepositoryAddress(store);
    if (err == KErrNone)
        {
        if (store == 0) { // Programming error in PSL, ptr/rc mismatch
 			err = KErrArgument; goto failed; }
        	
        iVariantStore = TRepositoryCompiled::New(reinterpret_cast<const HCR::SRepositoryCompiled *>(store));
        if (iVariantStore == 0) { 
			err = KErrNoMemory; goto failed; }

        }
    else if (err != KErrNotSupported)
    	goto failed;       
  
        
    // Ask the PSL if it wants the PIL not to search for the Core Image 
	// SRepositoryFile settings.
    iCoreImgStore = 0;
    if (!iVariant->IgnoreCoreImgRepository())
    	{
    	err = LocateCoreImgRepository(iCoreImgStore);
    	if (err == KErrNone)
     	   {
        	if (iCoreImgStore == 0) {
				err = KErrNoMemory; goto failed; }	
        	}
    	else if (err != KErrNotFound)
    		goto failed;
		}       
  
        
    // Ask the PSL for the address of the SRepositoryFile object. PSL 
    // can return KErrNotSupported & NULL if a local media based file 
	// repository is not used/support by PSL.  
    store = 0;
    err = iVariant->GetOverrideRepositoryAddress(store);
    if (err == KErrNone)
        {
        if (store == 0) { // Programming error in PSL, ptr/rc mismatch
 			err = KErrArgument; goto failed; }       
        
        iOverrideStore = TRepositoryFile::New(reinterpret_cast<const HCR::SRepositoryFile *>(store));
        if (iOverrideStore == 0) {
			err = KErrNoMemory; goto failed; }
			
        }
    else if (err != KErrNotSupported)
    	goto failed;       

	iStatus = EStatInitialised;
	
    // Sanity check here to ensure we have atleast one repository to use and run
    // sanity check on their contents to look for ordering issues and duplicates.
	HCR_TRACE3("=== HCR Ready: compiled:%x, coreimg:%x, override:%x", iVariantStore, iCoreImgStore, iOverrideStore);
    if ((iVariantStore == 0) && (iCoreImgStore == 0) && (iOverrideStore == 0)) {
 		err = KErrArgument; goto failed; }


#ifdef _DEBUG
	err = CheckIntegrity();
	if (err != KErrNone)
		goto failed;	
#endif

	iStatus = EStatReady;
	return KErrNone;

failed:
    iStatus = (iStatus & EStatMinorMask) | EStatFailed;
	HCR_TRACE_RETURN(err);
    }


TInt HCR::HCRInternal::SwitchRepository(const TText * aFileName, const TReposId aId)
	{
	HCR_FUNC("HCRInternal::SwitchRepository");
	
	TInt retVal = KErrNone;
	TRepository* store = NULL;

	if( aFileName != NULL)
		{
		retVal = SearchCoreImgRepository(store, aFileName);
		HCR_TRACE2("--- SearchCoreImgRepository()->%d (0x%08x)", retVal, retVal);
		}
		
	if( retVal == KErrNone )
		{
		switch(aId)
			{
			case ECoreRepos:
			    HCR_TRACE0("--- ECoreRepos");
				if( iCoreImgStore )
					{
					NKern::ThreadEnterCS();
					delete iCoreImgStore;
					NKern::ThreadLeaveCS();
					}
				iCoreImgStore = store;
				break;
				
			case EOverrideRepos:
			    HCR_TRACE0("--- EOverrideRepos");
				if( iOverrideStore )
    				{
	    			NKern::ThreadEnterCS();
					delete iOverrideStore;
					NKern::ThreadLeaveCS();
					}
				iOverrideStore = store;
				break;
		
			default:
			    HCR_TRACE0("--- default:");
				retVal = KErrNotSupported;
				break;		
			}
		}

	HCR_TRACE_RETURN(retVal);
    }

TInt HCR::HCRInternal::CheckIntegrity()
	{
	HCR_FUNC("HCRInternal::CheckIntegrity");
	
	TInt err = KErrNone;
	if (iVariantStore)
		{
		err = iVariantStore->CheckIntegrity();
		if (err != KErrNone)
			HCR_TRACEMSG_RETURN("HCR iVariantStore failed integrity check", err);
		}

	if (iCoreImgStore)
		{
		err = iCoreImgStore->CheckIntegrity();
		if (err != KErrNone)
			HCR_TRACEMSG_RETURN("HCR iCoreImgStore failed integrity check", err);
		}	
	
	if (iOverrideStore)
		{
		err = iOverrideStore->CheckIntegrity();
		if (err != KErrNone)
			HCR_TRACEMSG_RETURN("HCR iOverrideStore failed integrity check", err);
		}

	HCR_TRACE0("=== HCR Repository integrity checks PASSED!");
	return KErrNone;	
	}


TInt HCR::HCRInternal::FindSetting(const TSettingId& aId, TSettingType aType, 
        TSettingRef& aSetting)
    {
    HCR_FUNC("HCRInternal::FindSetting");
    TInt err = 0;
    TBool found = EFalse;
    
    HCR_TRACE3("--- Repository state: %x, %x, %x", iOverrideStore, iCoreImgStore, iVariantStore);
    
    if (iOverrideStore && 
        ((err = iOverrideStore->FindSetting(aId, aSetting)) == KErrNone))
        found = ETrue;
    if ((err != KErrNone) && (err != KErrNotFound))
        HCR_TRACE_RETURN(err);
        
    if (!found &&
        iCoreImgStore &&
        ((err = iCoreImgStore->FindSetting(aId, aSetting)) == KErrNone))
        found = ETrue;
    if ((err != KErrNone) && (err != KErrNotFound))
        HCR_TRACE_RETURN(err);

    if (!found &&
        iVariantStore &&
        ((err = iVariantStore->FindSetting(aId, aSetting)) == KErrNone))
        found = ETrue;
        
    if ((err != KErrNone) && (err != KErrNotFound))
        HCR_TRACE_RETURN(err);

    HCR_TRACE3("--- Search results: %d, %d, %x", found, err, aSetting.iSet);
    
    if (!found || (aSetting.iSet == 0))
        HCR_TRACE_RETURN(KErrNotFound);

    // Setting found at this point in the function
    //
    
    TSettingType type=static_cast<TSettingType>(aSetting.iRep->GetType(aSetting)); 
    if (type & ~aType)
        HCR_TRACE_RETURN(KErrArgument); // Wrong setting type
    
    HCR_TRACE3("--- Setting found! ID: (%d,%d) Type: %d", aId.iCat, aId.iKey, type);
    
    return KErrNone;
    }


TInt HCR::HCRInternal::FindSettingWithType(const TSettingId& aId, TSettingType& aType, 
      TSettingRef& aSetting)
    {
    HCR_FUNC("HCRInternal::FindSettingWithType");
    TInt err = 0;
    TBool found = EFalse;
    
    HCR_TRACE3("--- Repository state: %x, %x, %x", iOverrideStore, iCoreImgStore, iVariantStore);
    
    if (iOverrideStore && 
        ((err = iOverrideStore->FindSetting(aId, aSetting)) == KErrNone))
        found = ETrue;
    if ((err != KErrNone) && (err != KErrNotFound))
        HCR_TRACE_RETURN(err);
        
    if (!found &&
        iCoreImgStore &&
        ((err = iCoreImgStore->FindSetting(aId, aSetting)) == KErrNone))
        found = ETrue;
    if ((err != KErrNone) && (err != KErrNotFound))
        HCR_TRACE_RETURN(err);

    if (!found &&
        iVariantStore &&
        ((err = iVariantStore->FindSetting(aId, aSetting)) == KErrNone))
        found = ETrue;
        
    if ((err != KErrNone) && (err != KErrNotFound))
        HCR_TRACE_RETURN(err);

    HCR_TRACE3("--- Search results: %d, %d, %x", found, err, aSetting.iSet);
    
    if (!found || (aSetting.iSet == 0))
        {
        aType = ETypeUndefined;
        HCR_TRACE_RETURN(KErrNotFound);
        }

    // Setting found at this point in the function
    //
    
    aType=static_cast<TSettingType>(aSetting.iRep->GetType(aSetting)); 
    
    HCR_TRACE3("--- Setting found! ID: (%d,%d) Type: %d", aId.iCat, aId.iKey, aType);
    
    return KErrNone;
    }


TInt HCR::HCRInternal::GetWordSettings(TInt aNum, const SSettingId aIds[], 
        TInt32 aValues[], TSettingType aTypes[], TInt aErrors[])
    {
    HCR_FUNC("++ HCRInternal::GetWordSettings");
    HCR_TRACE3("--- Repository state: %x, %x, %x", iOverrideStore, iCoreImgStore, iVariantStore);
    
    if(aNum <= 0 || aIds == NULL || aErrors == NULL)
        HCR_TRACE_RETURN(KErrArgument);
    
    TInt err = 0;
    //If the user only supplies a single setting then there is no reasons to 
    //continue with multiple searach and it should be limited by internal 
    //invocation of FindSettingWithType.
    if(aNum == 1)
        {
        TSettingRef sref(0,0);
        TSettingType* pTypes;
                
        //aTypes array is optional and user may not provided it for us. So we
        //need to be sure it's not a null pointer
        if(aTypes == NULL)
            {
            //If this is a null pointer then just create our own element and 
            //assign it to the pTypes pointer
            TSettingType types[1];
            pTypes = types;
            }
        else
            {
            //else we use the user supplied array
            pTypes = aTypes;
            }
                
        //Let's find this setting
        err = HCRSingleton->FindSettingWithType(aIds[0], *pTypes, sref);
        
        //and analyse the result of operation
        
        //If setting is not found or it's larger than 4 bytes then store this
        //error cause in the user error array 
        if(err == KErrNotFound || err == KErrArgument)
            {
            //Indicate the error for the element and set the value to 0
            aErrors[0] = err;
            aValues[0] = 0;
            return 0;
            }
        //fatal error here, nothing to do, just exit and return the error code
        else if(err == KErrNotReady || err != KErrNone)
            {
            HCR_TRACE_RETURN(err);
            }
        else //err == KErrNone
            {
            //Get the value of the setting
            err = sref.iRep->GetValue(sref, reinterpret_cast<UValueWord&>(aValues[0]));

            //The GetValue can only return either KErrArgument or KErrNone
            if(err == KErrArgument)
                {
                aErrors[0] = KErrArgument;
                aValues[0] = 0;
                return 0;
                }
            else //err == KErrNone
                {
                aErrors[0] = KErrNone;
                }
            
            }
        
        //This single setting was found so indicate it to the user
        return (1);
        }

    
    //Introducing a SafeArray of pointers to the settings, which is passed to ver- 
    //sion of GetWordSettings() method declared in TRepository, and implemented 
    //in TRepositoryCompiled and TRepositoryFile
    TSa<SSettingId*> ids;

    //SafeArray of pointers to the aValues user array elements 
    TSa<TInt32*> values;
    
    //SafeArray of pointers to the aErrors user array elements 
    TSa<TInt*> errors;
    
    //SafeArray of pointers to the aTypes user array elements
    TSa<TSettingType*> types;
    
    
    //Local replacement for the aTypes[] array if it's not provided by user
    TSa<TSettingType> typesHolder;
    
    //Allocate the arrays of pointers in the  heap
    ids = new SSettingId*[aNum];
    values = new TInt32*[aNum];
    errors = new TInt*[aNum];
    types  = new TSettingType*[aNum];


    //Check all arrays allocations
    if(!ids() || !values() || !errors() || !types())
        {
        //One of the allocation was unsuccessful 
        HCR_TRACE_RETURN(KErrNoMemory);
        }
    
    //If the user did not supply the aTypes array for us we need to create one 
    //for ourself
    if(aTypes == NULL)
        {
        typesHolder = new TSettingType[aNum];
        if(!typesHolder())
            HCR_TRACE_RETURN(KErrNoMemory);
        }
    
       
    //Ininialize newly created array of pointers to the user supplied settings 
    for (TInt index = 0; index < aNum; index++)
        {
        ids[index] = const_cast<SSettingId*>(&aIds[index]);
        values[index] = const_cast<TInt32*>(&aValues[index]);
        errors[index] = &aErrors[index];
       
        if(aTypes == NULL)
            types[index] = &typesHolder[index];
        else
            types[index] = &aTypes[index];
        }
    
    
    //nfCount represents a total number of settings which were not found in all
    //repositories
    TInt nfCount = aNum;
    
    //nfReposCount represents a number of settings "not found - nf" in the searched
    //repository
    TInt nfReposCount   = 0;

    //It represents a number of setting found in the repository
    TInt reposCount   = 0;
    
    
    //First step through the Override store and gather all settings we need.
    //In the end of this procedure we'll have number of settings not found here
    //and found settings data are copied to the user arrays.
    if (iOverrideStore)
        {

        //Call the sibling method from the TRepositoryFile object
        err = iOverrideStore->GetWordSettings(aNum, ids(),
                values(), types(), errors());

        //Analyse the err we've got 
        if(err != KErrNone && err != KErrNotFound)
            {
            HCR_TRACE_RETURN(err);
            }
        else if(err == KErrNone)
            {
            //Search for number of not found parameters
            for(TInt index = 0; index < aNum; index ++)
                {
                switch(*(errors[index]))
                    {
                    //The setting was found or it's found but the type is larger
                    //than 4 bytes then we just increase a counter of the found
                    //settings in the repository
                    case KErrNone:
                    case KErrArgument:
                        reposCount ++;
                        break;


                    //The setting was not found, then re-initialize all the 
                    //arrays of pointers with the pointer to this element. 
                    //nfReposCount depict the counter of not found element and
                    //index shows the intial element position.
                    //As nfReposCount is always less or equal to index then we
                    //can easily make reassignment as nfReposCoun element was
                    //already analysed. In the end the nfReposCount is increased.
                    case KErrNotFound:
                        ids[nfReposCount]        = ids[index];
                        values[nfReposCount]     = values[index];
                        types[nfReposCount]      = types[index];
                        errors[nfReposCount]     = errors[index];                  
                        nfReposCount ++;
                        break;


                    default:
                        //No any action is needed
                        break;
                    }
                }

            }
        else //err == KErrNotFound
            {
            //No settings were found in the repository
            //reposCount is zero intialized, so nothing to do here
            }
        
        //Update the global counter only if there are some settings were found,
        //otherwise it can be situation when we overwrite the nfCount with zero
        //when either no any setting presents or no settings were found in the
        //repository
        if(reposCount > 0)
            nfCount = nfReposCount;
        }
    
    //Go through core image and search for the rest of settings
    nfReposCount = 0;
    reposCount = 0;
    
    if (iCoreImgStore && nfCount > 0)
        {

        err = iCoreImgStore->GetWordSettings(nfCount, ids(),
                                                values(), types(), errors());

        if (err != KErrNone && err != KErrNotFound)
            {
            HCR_TRACE_RETURN(err);
            }
        else if(err == KErrNone)
            {
            //Search for number of errors
            for(TInt index = 0; index < nfCount; index ++)
                {
                switch(*(errors[index]))
                    {
                    //The setting was found or it's found but the type is larger
                    //than 4 bytes then we just increase a counter of the found
                    //settings in the repository
                    case KErrNone:
                    case KErrArgument:
                        reposCount ++;
                        break;

                    //The setting was not found, then re-initialize all the 
                    //arrays of pointers with the pointer to this element. 
                    //nfReposCount depict the counter of not found element and
                    //index shows the intial element position.
                    //As nfReposCount is always less or equal to index then we
                    //can easily make reassignment as nfReposCoun element was
                    //already analysed. In the end the nfReposCount is increased.
                    case KErrNotFound:
                        ids[nfReposCount]        = ids[index];
                        values[nfReposCount]     = values[index];
                        types[nfReposCount]      = types[index];
                        errors[nfReposCount]     = errors[index];                  
                        nfReposCount ++;
                        break;


                    default:
                        //No any action is needed
                        break;

                    }

                }

            }
        else //err == KErrNotFound 
            {
            //No settings were found in the repository
            //reposCount is zero intialized, so nothing to do here
            }


        //Update the global counter only if there are some settings were found,
        //otherwise it can be situation when we overwrite the nfCount with zero
        //when either no any setting presents or no settings were found in the
        //repository 
        if(reposCount > 0)
            nfCount = nfReposCount;
        }
    
    //let's go through the last Variant store
    nfReposCount = 0;
    reposCount = 0;
    if(iVariantStore && nfCount > 0)
        {
        err = iVariantStore->GetWordSettings(nfCount, ids(), values(), 
                types(), errors());

        if (err != KErrNone && err != KErrNotFound)
            {
            HCR_TRACE_RETURN(err);
            }
        else if(err == KErrNone)
            {
            //Search for number of errors
            for(TInt index = 0; index < nfCount; index ++)
                {
                switch(*(errors[index]))
                    {
                    //The setting was found or it's found but the type is larger
                    //than 4 bytes then we just increase a counter of the found
                    //settings in the repository
                    case KErrNone:
                    case KErrArgument:
                        reposCount ++;
                        break;

                    //The setting was not found, then re-initialize all the 
                    //arrays of pointers with the pointer to this element. 
                    //nfReposCount depict the counter of not found element and
                    //index shows the intial element position.
                    //As nfReposCount is always less or equal to index then we
                    //can easily make reassignment as nfReposCoun element was
                    //already analysed. In the end the nfReposCount is increased.
                    case KErrNotFound:
                        *values[nfReposCount]     = 0;
                        *types[nfReposCount]      = ETypeUndefined;
                        *errors[nfReposCount]     = KErrNotFound;
                        nfReposCount ++;
                        break;


                    default:
                        //No any action is needed
                        break;

                    }
                }
            
            }
        else //err == KErrNotFound
            {
            //No settings were found in the repository
            //reposCount is zero intialized, so nothing to do here
            }
        
        //Update the global counter only if there are some settings were found,
        //otherwise it can be situation when we overwrite the nfCount with zero
        //when either no any setting presents or no settings were found in the
        //repository
        if(reposCount > 0)
            nfCount = nfReposCount;
        }
    //Return the number of found elements
    return (aNum - nfCount);
    }





TInt HCR::HCRInternal::FindNumSettingsInCategory (TCategoryUid aCatUid)
    {
    HCR_FUNC("++ HCRInternal::FindNumSettingsInCategory");
    TInt err = 0;

    HCR_TRACE3("--- Repository state: %x, %x, %x", iOverrideStore, iCoreImgStore, iVariantStore);

    //First and last element index within category in the Override store
    TInt32 oLowIndex = 0;
    TInt32 oHighIndex = 0;
    TInt oCount = 0;

    //Find numOverride number of settings within the category in the OverrideStore 
    //repository
    if(iOverrideStore)
        {
        err = iOverrideStore->FindNumSettingsInCategory(aCatUid, 
                oLowIndex, oHighIndex);
       
        if(err == KErrNotFound)
            oCount = 0;
        else
            oCount = oHighIndex - oLowIndex + 1;

        //If CoreImg and Variant store are not activated so just return the
        //number of elements found in the Override store
        if(!iCoreImgStore && !iVariantStore)
            return oCount;
        }


    //First and last element index within category in the CoreImg store
    TInt32 cLowIndex = 0;
    TInt32 cHighIndex = 0;
    TInt32 cLength = 0;
    TInt   cCount = 0;

        
    
    //Temproary holder for the found element position
    TInt32 elementPos;
    //Temproary holder for the low index, which is used to decrease the scope
    //of search
    TInt32 lowIndex = oLowIndex;
    
    //Setting data holders
    SSettingId setId;
    TSettingRef setRef;
    
    if(iCoreImgStore)
        {
        //Find numCoreImg number of settings within the category in the CoreImg re-
        //pository
        err = iCoreImgStore->FindNumSettingsInCategory(aCatUid, 
                cLowIndex, cHighIndex);

        if(err == KErrNotFound)
            cLength = 0;
        else
            //Calculate the number of elements within category, in CoreImg store
            cLength = cHighIndex - cLowIndex + 1;

        if(oCount > 0)
            {
            //Find all elemnts from CoreImg store which are not redefined in the 
            //Override store. When element is not found in the Override store 
            //then cCount is increased.
            for(TInt element = 0; element < cLength; element ++)
                {
                //Find element in the repository by its index
                iCoreImgStore->GetSettingRef(cLowIndex + element, setRef);
                //and get its id
                iCoreImgStore->GetId(setRef, setId);
                
                //Check either this element is already redefined in the Override
                //store
                err = iOverrideStore->FindSetting( setId, setRef, 
                        elementPos, lowIndex, oHighIndex);

                if(err == KErrNone)
                    {
                    //if the element is found in the Override store, then store the posi-
                    //tion of this element in lowIndex, to narrow next search procedure
                    lowIndex = elementPos;
                    }
                else if(err == KErrNotFound)
                    {
                    //if element is not found then it means it's not redefined in the 
                    //Override store and this element must be counted in the total number
                    //of elemnts in all stores
                    cCount ++;
                    }
                else
                    {
                    return err;
                    }
                }
            }
        else
            {
            cCount = cLength;
            }



        //Check if the Variant store is present if it's not then just return the
        //result
        if(!iVariantStore)
            return (oCount + cCount);

        }

    //First and last element index within giving category in the Variant store
    TInt32 vLowIndex  = 0;
    TInt32 vHighIndex = 0;
    TInt32 vLength = 0;
    TInt vCount = 0;

    if(iVariantStore)
        {
        //Find numVariant number of settings within the category in the VariantStore
        //repository
        err = iVariantStore->FindNumSettingsInCategory(aCatUid, vLowIndex, 
                vHighIndex);

        //Analyze returned error code

        if(err == KErrNotFound)
            vLength = 0;
        else
            //Calculate the number of elements within category, in CoreImg store
            vLength = vHighIndex - vLowIndex + 1;

        if(oCount == 0 && cCount == 0)
            {
            return vLength;
            }

        if(oCount > 0 || cCount >0)
            {
            //Find all elemnts from Variant store which are not redefined either in the 
            //Override or CoreImg store. These elements are added to the total 
            //count.
            
            // Some additional containers. They are needed because we  
            // must check two stores Override and Variant in this iteration. Making a 
            // decision of uniqueness of the element is made from the analyse of both  
            // result. The element is only unique defined in the Variant store if it's  
            // not redefined either in the Override or Variant store
            TSettingRef tmpRef;
            //Temproary holder for the found element position
            TInt32 elementPos2 = 0;
            //Temproary holder for the low index, which is used to decrease the scope
            //of search
            TInt32 lowIndex2 = cLowIndex;
            // This index contains Override low index and will be changed by the position
            // of a new found element 
            lowIndex= oLowIndex;

            TBool isRedefined = EFalse;
            
            for(TInt element = 0; element < vLength; element ++)
                {
                //Find the setting in the repository by its index and
                iVariantStore->GetSettingRef(vLowIndex + element, setRef);
                //get its id
                iVariantStore->GetId(setRef, setId);

                if(oCount > 0)
                    {
                    //Check either this element is already redefined in the Override store 
                    err = iOverrideStore->FindSetting(setId, tmpRef, 
                            elementPos, lowIndex, oHighIndex);

                    if(err == KErrNone)
                        {
                        //if the element is found in the Override store, then store the posi-
                        //tion of this element in lowIndex, to narrow next search procedure
                        lowIndex = elementPos;
                        isRedefined = ETrue;
                        }
                    else if(err == KErrNotFound)
                        {
                        //the element is not presented in the Override store
                        //nothing to do here
                        }
                    else
                        {
                        return err;
                        }

                    }


                if(cCount > 0 && !isRedefined)
                    {
                    //Check either this element is already redefined in the CoreImg store
                    err = iCoreImgStore->FindSetting(setId, tmpRef, 
                            elementPos2, lowIndex2, cHighIndex);


                    if(err == KErrNone)
                        {
                        //if the element is found in the Override store, then store the posi-
                        //tion of this element in lowIndex, to narrow next search procedure
                        lowIndex2 = elementPos2;
                        isRedefined = ETrue;
                        }
                    else if(err == KErrNotFound)
                        {
                        //the element is not presented in the Override store
                                                //nothing to do here
                        }
                    else
                        {
                        return err;
                        }
                    }
                

                if(!isRedefined)
                    vCount ++;
                else
                    isRedefined = EFalse;

                }//for(TInt element = 0; element < vLength; element ++)

            }
        else
            {
            vCount = vLength;
            }
        }

    //Return the total number of elements found in the category
    return (oCount + cCount + vCount);
    }




TInt HCR::HCRInternal::FindSettings(TCategoryUid aCatUid, 
        TInt aMaxNum, TElementId aIds[],  
        TSettingType aTypes[], TUint16 aLens[])
    {
    HCR_FUNC("++ HCRInternal::FindSettings w/o patterns");
   
    HCR_TRACE3("--- Repository state: %x, %x, %x", iOverrideStore, 
            iCoreImgStore, iVariantStore);
   
    //Error container
    TInt err = KErrNone;

    //Total number of found elements
    TInt numFound = 0;
    //Number of found elements in the Override store
    TInt oNumFound = 0;

    //Low and High indexes in the Override store
    TInt32 oLoIndex = 0;
    TInt32 oHiIndex = 0;

    //Temproary holder for the found element position
    TInt32 elementPos = 0;
    TInt32 lowIndex = 0;
    

    //Tempoary type and length value holders if the
    //user does not provide these arrays for us
    TSettingType tmpType;
    TUint16 tmpLen;

        
    //Setting datat holders
    TSettingRef setRef;
    TSettingId  setId;

    
    //Find number of elements, low and hingh index in the Override store
    if(iOverrideStore)
        {
        err = iOverrideStore->FindNumSettingsInCategory(aCatUid, oLoIndex,
                oHiIndex);
        if(err == KErrNone)
            {
            //If number of elements in the Override Store is larger than aMaxNum or 
            //CoreImage/Variant stores are not present then write all found 
            //settings into the user array, return the number of found elements and
            //exit
            oNumFound = (oHiIndex - oLoIndex + 1);
            lowIndex = oLoIndex;
       
            if(oNumFound < aMaxNum)
                {
                for(TInt index = 0; index < oNumFound; index ++)
                    {
                    //Get setting reference data from the repository
                    iOverrideStore->GetSettingRef(oLoIndex + index, setRef);

                    //Copy the settings data into the user arrays
                    iOverrideStore->GetSettingInfo(setRef, 
                            aIds[index], 
                            aTypes ? aTypes[index]:tmpType,
                            aLens ? aLens[index]:tmpLen);

                    
                    }
                }
            else //oNumFound >= aMaxNum
                {
                //Copy data to the user array
                for(TInt index = 0; index < aMaxNum; index++)
                    {
                    //Get setting reference data from the repository
                    iOverrideStore->GetSettingRef(oLoIndex + index, setRef);
                    //Copy the settings data into the user arrays
                    iOverrideStore->GetSettingInfo(setRef, 
                            aIds[index], 
                            aTypes ? aTypes[index]:tmpType,
                            aLens  ? aLens[index]:tmpLen);

                    }
                return oNumFound;
                }
            }
        else if(err == KErrNotFound)
            {
            //Nothing to do here, oNumFound is set to zero already
            }
        else 
            {
            return err;
            }
        }

   
    //Low/High index in the CoreImg
    TInt32 cLoIndex = 0;
    TInt32 cHiIndex = 0;
    TInt cNumFound = 0;
    
    //Temproary setting reference holder
    TSettingRef tmpRef;

    //Temproary holder for the found element position
    elementPos = 0;
    lowIndex = oLoIndex;

    //Redefined status flag, it's used to flag that the element is found in the 
    //upper stores
    TBool isRedefined = EFalse;
    
    //User array index
    TInt usrArrIndx = 0;

    //If the count is still less than aMaxNum then continue with searching 
    //settings in the CoreImage store
    if(iCoreImgStore)
        {

        //Find number of elements and low/high indexes
        err = iCoreImgStore->FindNumSettingsInCategory(aCatUid, cLoIndex,
                cHiIndex);

        if(err == KErrNone)
            {
            for(TInt index = 0; index < (cHiIndex - cLoIndex + 1); index ++)
                {
                //Get the setting data by its index in the repository
                iCoreImgStore->GetSettingRef(cLoIndex + index, setRef);
                //get setting id
                iCoreImgStore->GetId(setRef, setId);
                
                if(oNumFound > 0)
                    {
                    //Check either this element is already redefined in the 
                    err = iOverrideStore->FindSetting(setId, tmpRef, 
                            elementPos, lowIndex, oHiIndex);

                    
                    if(err == KErrNone)
                        {
                        lowIndex = elementPos + 1;
                        isRedefined = ETrue;
                        }
                    else if (err == KErrNotFound)
                        {
                        //Nothing to do hear, isRedefined flag is EFalse
                        //all analysis is done later in the code
                        }
                    else
                        {
                        return err;
                        }
                    }

                //Examine the redefined status flag
                if(!isRedefined)
                    {
                    // If the element was not found then we need to copy to 
                    // the pA array and increase the counter of setting data 
                    // only if we did not reach the aMaxNum of found elements
                    
                    usrArrIndx = oNumFound + cNumFound;
                    if(usrArrIndx < aMaxNum)
                        {
                        //Copy the settings data into the user arrays
                        iCoreImgStore->GetSettingInfo(setRef, 
                                 aIds[usrArrIndx], 
                                 aTypes ? aTypes[usrArrIndx]:tmpType,
                                 aLens ? aLens[usrArrIndx]:tmpLen);
                        cNumFound ++;
                        }
                    else
                        {
                        //All required elements were found!
                        //As vNumFound was used an index so it's value it runs 
                        //from 0 to X-1, where X - number of found elements. To 
                        //get number of elements found we need increase the last
                        //counter value by 1.
                        if(cNumFound != 0)
                            cNumFound ++;
                        //It reaches the goal, all required elements are found
                        //stop here and return the result
                        break;
                        }
                    }
                else
                    //Element is found in other repositories, just reset a flag
                    isRedefined = EFalse;
                }
            }
        else if (err == KErrNotFound)
            {
            //cNumFound is already set to zero during the initialization
            //Nothing to do here
            }
        else //any other errors
            {
            return err;
            }
        }

    
    //Low/High index in the CoreImg
    TInt32 vLoIndex = 0;
    TInt32 vHiIndex = 0;
    TInt vNumFound = 0;

    //Temproary holder for the found element position
    TInt32 elementPos2 = 0;
    
    TInt32 lowIndex2 = cLoIndex;
    lowIndex  = oLoIndex;

    isRedefined = EFalse;
    

    //If the count is still less than aMaxNum then continue with searching 
    //settings in the CoreImage store
    if(iVariantStore)
        {

        //Find number of elements and low/high indexes
        err = iVariantStore->FindNumSettingsInCategory(aCatUid, vLoIndex,
                vHiIndex);
        if(err == KErrNone)
            {

            for(TInt index = 0; index < (vHiIndex - vLoIndex + 1); index ++)
                {
                //Get setting reference data by its index in the repository
                iVariantStore->GetSettingRef(vLoIndex + index, setRef);
                
                //and get setting id
                iVariantStore->GetId(setRef, setId);
                
                if(oNumFound > 0)
                    {
                    //Check either this element is already redefined in the 
                    err = iOverrideStore->FindSetting(setId, tmpRef, elementPos,  
                            lowIndex, oHiIndex);
                    
                
                    //Also suppress the error checking due the reason described 
                    //above
                    if(err == KErrNone)
                        {
                        lowIndex = elementPos + 1;
                        isRedefined = ETrue;
                        }
                    else if (err == KErrNotFound)
                        {
                        //Element is not found, nothing to proceed here
                        }
                    else
                        {
                        return err;
                        }
                    }

                if(cNumFound > 0 && !isRedefined)
                    {
                    //Check either this element is already redefined in the 
                    err = iCoreImgStore->FindSetting(setId, tmpRef, elementPos2,  
                            lowIndex2, cHiIndex);

                    if(err == KErrNone)
                        {
                        lowIndex2 = elementPos2 + 1;
                        isRedefined = ETrue;
                        }
                    else if (err == KErrNotFound)
                        {
                        //Element is not found, nothing to proceed here
                        }
                    else
                        {
                        return err;
                        }
                    }
               
                if(!isRedefined)
                    {
                    usrArrIndx = oNumFound + cNumFound + vNumFound;
                    if(usrArrIndx < aMaxNum)
                        {
                        //Copy the settings data into the user arrays
                        iVariantStore->GetSettingInfo(setRef, 
                                 aIds[usrArrIndx], 
                                 aTypes ? aTypes[usrArrIndx]:tmpType,
                                 aLens ? aLens[usrArrIndx]:tmpLen);

                        vNumFound ++;
                        }
                    else
                        {
                        //All required elements were found!
                        //As vNumFound was used an index so it's value it runs 
                        //from 0 to X-1, where X - number of found elements. To 
                        //get number of elements found we need increase the last
                        //counter value by 1.
                        if(vNumFound != 0)
                            vNumFound ++;
                        //It reaches the goal, all required elements are found
                        //stop here and return the result
                        break;
                        }
                    }
                else
                    {
                    isRedefined = EFalse;
                    }
                }
            }
        else if (err == KErrNotFound)
            {
            //oNumFound is already set to zero during the initialization
            //Nothing to do here
            }
        else
            {
            return err;
            }
        
        }
    
    //Let's prepare the final data
    numFound = oNumFound + cNumFound + vNumFound;

    //Return result to the user
    return numFound;
    }







TInt HCR::HCRInternal::FindSettings(TCategoryUid aCat, TInt aMaxNum,
                TUint32 aMask, TUint32 aPattern, 
                TElementId aIds[], TSettingType aTypes[], TUint16 aLens[])
    {
    //Holder for errors and number of elements
    TInt r = KErrNone;
    //Total number of elements within the given category
    TInt allInCatFound = 0;
    //Number of elements which corresponds to the aMask and aPattern
    TInt numFound = 0;
    
    //Find the number of elements within the category
    r = FindNumSettingsInCategory(aCat);
    
    //Analyse the returned error
    //if r < 0 - this is an error return to the user 
    //if r > 0 - number of found settings
    if(r < 0)
        {
        HCR_TRACE_RETURN(r);
        }
    else if (r == 0)
        //No any elements found for this category 
        return 0;
    else
        allInCatFound = r;
    
    //Result data array holder
    TSa<TElementId> pIds; 
    TSa<TSettingType> pTypes;
    TSa<TUint16> pLens;
    
    pIds = new TElementId[allInCatFound];
    pTypes = new TSettingType[allInCatFound];
    pLens = new TUint16[allInCatFound];

    if(pIds() == NULL || pTypes() == NULL || pLens() == NULL)
        //One of the allocation was unsuccessful 
        HCR_TRACE_RETURN(KErrNoMemory);
    
    r = FindSettings(aCat, allInCatFound, pIds(), pTypes(), pLens());
    
    //Exit if we've got negative result just report error
    if(r < 0)
        HCR_TRACE_RETURN(r);
    
    //Somehow we'got less elements than it must be!!!
    __NK_ASSERT_DEBUG(r == allInCatFound);
    if(r < allInCatFound)
        HCR_TRACE_RETURN(KErrGeneral);
    
    
    //Choose the elements which satisfy this condition
    //((elementID & aElementMask) == (aPattern & aElementMask)). The total num-
    //ber of returned elements should not exceed the aMaxNum
    for(TInt index = 0; index < allInCatFound; index++)
        {
            if(((pIds[index] & aMask) == (aPattern & aMask)))
                {
                aIds[numFound] = pIds[index];

                if(aTypes)
                    aTypes[numFound] = pTypes[index];

                if(aLens)
                    aLens[numFound] = pLens[index];

                numFound ++;
                }
            else
                continue;
            
            //Check either we already found  or not enough elements
            //If we did then break the loop
            if(numFound == aMaxNum)
                break;
        }
    
    return numFound;
    }


// -- METHODS -----------------------------------------------------------------
//
// TRepository

HCR::TRepository::~TRepository()
	{
    HCR_FUNC("~TRepository");
	}

TBool HCR::TRepository::IsWordValue(const TSettingRef& aRef)
    {
    HCR_FUNC("TRepository::IsWordValue");
    return ((aRef.iSet->iType & KMaskWordTypes) != 0);
    }

TBool HCR::TRepository::IsLargeValue(const TSettingRef& aRef)
    {
    HCR_FUNC("TRepository::IsLargeValue");
    return ((aRef.iSet->iType & KMaskLargeTypes) != 0);
    }

void HCR::TRepository::GetId(const TSettingRef& aRef, TCategoryUid& aCat, TElementId& aKey)
    {
    HCR_FUNC("TRepository::GetId1");
    aCat = aRef.iSet->iId.iCat;
    aKey = aRef.iSet->iId.iKey;
    }

void HCR::TRepository::GetId(const TSettingRef& aRef, SSettingId& aId)
    {
    HCR_FUNC("TRepository::GetId2");
    aId = aRef.iSet->iId;
    }

TInt32 HCR::TRepository::GetType(const TSettingRef& aRef)
    {
    HCR_FUNC("TRepository::GetType");
    return (aRef.iSet->iType);
    }

TUint16 HCR::TRepository::GetLength(const TSettingRef& aRef)
    {
    HCR_FUNC("TRepository::GetLength");
    
	// Assume large value, will be caught when value retreived if not correct.
	// Saves some CPU cycles...
	// if (IsLargeValue(aRef))
    return (aRef.iSet->iLen);
    }

void HCR::TRepository::GetSettingInfo(const HCR::TSettingRef& aSetRef, 
               HCR::TElementId& aId, HCR::TSettingType& aType, TUint16& aLen)
    {
    HCR_FUNC("TRepository::GetSettingInfo");

    aId = aSetRef.iSet->iId.iKey;
   
    aType = static_cast<TSettingType>(aSetRef.iSet->iType);

    aLen = aSetRef.iSet->iLen;
    }

// -- METHODS -----------------------------------------------------------------
//
// TRepositoryCompiled


HCR::TRepository* HCR::TRepositoryCompiled::New(const SRepositoryCompiled* aRepos)
    {
    HCR_FUNC("TRepositoryCompiled::New");
    
    __NK_ASSERT_ALWAYS(aRepos != 0);
    return new TRepositoryCompiled(aRepos);
    }

HCR::TRepositoryCompiled::TRepositoryCompiled(const SRepositoryCompiled* aRepos)
 : iRepos(aRepos)
    {
    HCR_FUNC("TRepositoryCompiled");
    }
    
HCR::TRepositoryCompiled::~TRepositoryCompiled()
    {
    HCR_FUNC("~TRepositoryCompiled");
    }
        
TInt HCR::TRepositoryCompiled::CheckIntegrity()
    {
    HCR_FUNC("TRepositoryCompiled::CheckIntegrity");
    
    __NK_ASSERT_ALWAYS(this != 0);   
    __NK_ASSERT_ALWAYS(iRepos != 0);

	if (iRepos->iOrderedSettingList == 0)
        HCR_TRACEMSG_RETURN("Compiled Repository header missing setting array list", KErrNotFound);
    
	HCR_TRACE2("Compiled repository 0x%x contains %05d entries", iRepos, iRepos->iHdr->iNumSettings);

    SSettingC* arr = iRepos->iOrderedSettingList;
    TSettingId prev(0,0);
    TInt rc=0;
    for (int i=0; i < iRepos->iHdr->iNumSettings; i++, arr++)
    	{
    	__NK_ASSERT_ALWAYS(arr != 0);
    	HCR_TRACE3("Checking entry %05d - (0x%x,0x%x)", i, arr->iName.iId.iCat,  arr->iName.iId.iKey);
    	rc = CompareSSettingIds(prev, arr->iName.iId);
		// Check for duplicates that reside next to each other
    	if ((i > 0) && (rc == 0))
    		HCR_TRACE_RETURN (KErrAlreadyExists);
    	// Check that the entries are in ascending order	
    	if (rc != -1)
    		HCR_TRACE_RETURN (KErrCorrupt);
    	prev = arr->iName.iId;
		}
    return KErrNone; 
    }
    
TInt HCR::TRepositoryCompiled::FindSetting(const TSettingId& aId, TSettingRef& aSetting)
    {
    HCR_FUNC("TRepositoryCompiled::FindSetting");
    
    __NK_ASSERT_DEBUG(iRepos != 0);
    __NK_ASSERT_DEBUG(iRepos->iHdr != 0);
    
    if ((iRepos->iHdr->iNumSettings == 0) || 
        (iRepos->iOrderedSettingList == 0))
        HCR_TRACE_RETURN(KErrNotFound);
    
    SSettingC* arr = iRepos->iOrderedSettingList;
    int low = 0;
    int high = iRepos->iHdr->iNumSettings-1;
    int mid;
    int com;
    
    while (low<=high)
        {
        mid = (low+high) >> 1;
        com = CompareSSettingIds(aId, arr[mid].iName.iId);
        if (com < 0)
            high = mid-1;
        else if (com > 0)
            low = mid+1;
        else
            {
            aSetting.iRep = this;
            aSetting.iSet = &((arr[mid]).iName);
            return KErrNone;
            }    
        } 
        
    aSetting.iRep = 0;
	aSetting.iSet = 0; 
    return KErrNotFound;
    }




TInt HCR::TRepositoryCompiled::FindSetting(const TSettingId& aId, 
       TSettingRef& aSetting,  TInt32& aPosition,  TInt32 aLow, TInt32 aHigh)
    {
    HCR_FUNC("TRepositoryCompiled::FindSetting within the given range");
    
    
    __NK_ASSERT_DEBUG(iRepos != 0);
    __NK_ASSERT_DEBUG(iRepos->iHdr != 0);
    
    if ((iRepos->iHdr->iNumSettings == 0) || 
        (iRepos->iOrderedSettingList == 0))
        HCR_TRACE_RETURN(KErrNotFound);
    
    SSettingC* arr = iRepos->iOrderedSettingList;
    TInt32 low = aLow;
    TInt32 high = aHigh;
    TInt32 mid;
    TInt32 com;
    
    while (low<=high)
        {
        mid = (low+high) >> 1;
        com = CompareSSettingIds(aId, arr[mid].iName.iId);
        if (com < 0)
            high = mid-1;
        else if (com > 0)
            low = mid+1;
        else
            {
            aSetting.iRep = this;
            aSetting.iSet = &((arr[mid]).iName);
            aPosition = mid;
            return KErrNone;
            }    
        } 
        
    aSetting.iRep = 0;
    aSetting.iSet = 0;
    aPosition = 0;
    return KErrNotFound;
    }
    


TInt HCR::TRepositoryCompiled::GetWordSettings(TInt aNum,   
       SSettingId* aIds[], TInt32* aValues[], TSettingType* aTypes[],
        TInt* aErrors[])
    {
    HCR_FUNC("TRepositoryCompiled::GetWordSettings");
    
    __NK_ASSERT_DEBUG(iRepos != 0);
    __NK_ASSERT_DEBUG(iRepos->iHdr != 0);
    
    if ((iRepos->iHdr->iNumSettings == 0) || 
        (iRepos->iOrderedSettingList == 0))
        HCR_TRACE_RETURN(KErrNotFound);
 
    TInt err = KErrNone;
        
    TInt32 rMaxIndex = 0;
    TInt32 rMinIndex = 0;
    TInt32 uFirstIndex = 0;
    TInt32 uLastIndex = 0;
    TInt32 rIndex = 0;
    TInt32 uIndex = 0;
    
    TSettingRef settingRef(NULL, NULL);
    SSettingC* pSetting = NULL;


    //Find position index within the repository for the first and last setting
    //from user supplied array aIds[]
    uIndex = 0;
    TBool isRedefined = EFalse;
    err = KErrNotFound;
    uFirstIndex = 0;
    while(!isRedefined && uIndex < aNum)
        {
        //Find first setting from user array. The importance here is that we   
        //should get value of first setting index in the repository in rMinIndex.
        //This time the  scope of search is whole repository.
        err = this->FindSetting(*aIds[uIndex],settingRef, rMinIndex, 
                0, iRepos->iHdr->iNumSettings);
        if(err == KErrNotFound)
            {
            *aErrors[uIndex] = err;
            *aValues[uIndex] = 0;
            //Copy type only if user provided aTypes array
            if(aTypes)
                *aTypes[uIndex] = ETypeUndefined;
            
            //As FindSetting did not find the element, let's challenge with 
            //the next one from aIds[] array
            uIndex ++;
            continue;
            }
        //fatal error here, nothing to do, just exit and return the error code
        else if(err == KErrNotReady || err != KErrNone)
            {
            return err;
            }
        else // err == KErrNone
            {
            //Get the value and type
            pSetting = (SSettingC*) settingRef.iSet;
            //again copy the type value into the user array if it's provided
            if(aTypes)
                *aTypes[uIndex] = static_cast<TSettingType>(settingRef.iSet->iType); 

            //Check for the found type is this word size? If it's not then 
            //indicate error for this setting
            if(*aTypes[uIndex] > ETypeLinAddr)
                {
                *aErrors[uIndex] = KErrArgument;
                *aValues[uIndex] = 0;
                }
            else
                {
                *aErrors[uIndex] = KErrNone;
                *aValues[uIndex] = pSetting->iValue.iLit.iInt32;
                }
            
            //Break the loop by setting the redefined status
            isRedefined = ETrue;
            }
        }
    
    //At this point we should find at least one element from the user array,   
    //store this index in the local variable, it is used later in the code.   
    //Please be noticed we've also got rMinIndex - first setting index in the
    //repository.
    if(err == KErrNone)
        uFirstIndex = uIndex;
    //if we did not find any elements at all just return KErrNotFound
    else
        return KErrNotFound;

    
    
    //Now lets find the last setting
    uIndex = aNum - 1;
    isRedefined = EFalse;
    err = KErrNotFound;
    while(!isRedefined && uIndex > uFirstIndex)
        {
        //Find the last setting from user array. The importance here is that we   
        //should get value of first setting index in the repository in 
        //rMinIndex. This time the  scope of search is whole repository.
        err = this->FindSetting(*aIds[uIndex],settingRef, rMaxIndex, 
                rMinIndex, iRepos->iHdr->iNumSettings);
        if(err == KErrNotFound)
            {
            *aErrors[uIndex] = err;
            *aValues[uIndex] = 0;
            if(aTypes)
                *aTypes[uIndex] = ETypeUndefined;
            
            //As FindSetting did not find the element, let's challenge with 
            //previous one, as we are moving in reverse direction
            uIndex --;
            continue;
            }
        //fatal error here, nothing to do, just exit and return the error code
        else if(err == KErrNotReady || err != KErrNone)
            {
            return err;
            }
        else //err == KErrNone
            {
            pSetting = (SSettingC*) settingRef.iSet;
            if(aTypes)
                *aTypes[uIndex] = static_cast<TSettingType>(settingRef.iSet->iType); 
            
            //Check for the found type is this word size? If it's not then indicate
            //error for this setting
            if(*aTypes[uIndex] > ETypeLinAddr)
                {
                *aErrors[uIndex] = KErrArgument;
                *aValues[uIndex] = 0;
                }
            else
                {
                *aErrors[uIndex] = KErrNone;
                *aValues[uIndex] = pSetting->iValue.iLit.iInt32;
                }

            isRedefined = ETrue;
            }
        }

    //At this point we found the last setting, store it's user array index in   
    //the local variable, it is used later in the code. Please be noticed   
    //we've also got rMaxIndex - last setting index in the repository.
    if(err == KErrNone)
        uLastIndex = uIndex;
    else
        //if we are here we did not find any other elements than was found
        //in previous iteration then just stop here
        return KErrNotFound;
    
    //The scope of user array settings in the repository is found. 
    //Let's find all other settings from user array. Bare in mind the low
    //bound for the repository index is increased each iteration to optimize the
    //search time.
    for(uIndex = uFirstIndex + 1; uIndex < uLastIndex; uIndex ++)
        {
        err = this->FindSetting(*aIds[uIndex],settingRef, rIndex, 
                rMinIndex, rMaxIndex);
        if(err == KErrNotFound)
            {
            *aErrors[uIndex] = err;
            *aValues[uIndex] = 0;
            if(aTypes)
                *aTypes[uIndex] = ETypeUndefined;

            //As FindSetting did not find the element, let's challenge with 
            //another one
            continue;
            }
        else if(err == KErrNotReady || err != KErrNone)
            {
            return err;
            }
        else //err == KErrNone
            {

            pSetting = (SSettingC*) settingRef.iSet;
            if(aTypes)
                *aTypes[uIndex] = static_cast<TSettingType>(settingRef.iSet->iType); 

            //Check for the found type is this word size? If it's not then indicate
            //error for this setting
            if(*aTypes[uIndex] > ETypeLinAddr)
                {
                *aErrors[uIndex] = KErrArgument;
                *aValues[uIndex] = 0;
                }
            else
                {
                *aErrors[uIndex] = KErrNone;
                *aValues[uIndex] = pSetting->iValue.iLit.iInt32;
                }

            rMinIndex = rIndex + 1;

            }

        }

    return KErrNone;
    }





TInt HCR::TRepositoryCompiled::FindNumSettingsInCategory(TCategoryUid aCatUid,
        TInt32& aFirst, TInt32& aLast)
    {
    HCR_FUNC("TRepositoryCompiled::FindNumSettingsInCategory");

    __NK_ASSERT_DEBUG(iRepos != 0);
    __NK_ASSERT_DEBUG(iRepos->iHdr != 0);

    if ((iRepos->iHdr->iNumSettings == 0) || (iRepos->iOrderedSettingList == 0))
        HCR_TRACE_RETURN(KErrNotFound);

    SSettingC* arr = iRepos->iOrderedSettingList;
    int low = 0;
    int high = iRepos->iHdr->iNumSettings-1;
    int mid = 0;
    int com = 0;

    //Let's find any setting within the category, mid will store the setting 
    //index in the repository
    while (low<=high)
        {
        mid = (low+high) >> 1;
        com = CompareByCategory(aCatUid, arr[mid].iName.iId);
        if (com < 0)
            high = mid-1;
        else if (com > 0)
            low = mid+1;
        else
            {
            break;
            }    
        } 

    // If no one setting with the given category was found the return error  
    // to the user
    if(low > high)
        {
        aFirst = 0;
        aLast  = 0;
        return KErrNotFound;
        }

    //Search the first element within the category
    low = mid;
    while(low > 0 && arr[low].iName.iId.iCat == aCatUid)
        {
        low --;
        }
    //Check the boundary conditions, there are two cases when we exit the loop
    //either we found an element which category is not one we are looking for or
    //we reach the beggining of the repository. If we reach the beggining of the
    //repository we don't really know is it because this is last elment or it
    //has required aCatUid, so we check these two conditions below
    if(low == 0 && arr[low].iName.iId.iCat == aCatUid)
        aFirst = low;
    //We finish the loop either reaching the setting which category id is not
    //what we need or this is first setting in the repository again with another
    //category, so in both case we throw this element from the account.
    else
        aFirst = low + 1;

    //Search the last element within the category
    high = mid;
    while(high < iRepos->iHdr->iNumSettings && arr[high].iName.iId.iCat == aCatUid)
        {
        high ++;
        }
    
    //Same situation as above, boundary conditions
    if(high == (iRepos->iHdr->iNumSettings -1) && arr[high].iName.iId.iCat == aCatUid)
        aLast = high;
    else
        aLast = high -1;


    return KErrNone;
    }



void HCR::TRepositoryCompiled::GetSettingRef(TInt32 aIndex, 
        HCR::TSettingRef& aRef)
    {
    __NK_ASSERT_DEBUG(iRepos != 0);
    __NK_ASSERT_DEBUG(iRepos->iHdr != 0);
    __NK_ASSERT_DEBUG(aIndex >=0 && aIndex < iRepos->iHdr->iNumSettings);
    
    if ((iRepos->iHdr->iNumSettings == 0) || (iRepos->iOrderedSettingList == 0))
        {
        aRef.iRep = NULL;
        aRef.iSet = NULL;
        }

    //Get the pointer to the repository data
    SSettingC* arr = iRepos->iOrderedSettingList;
        
    aRef.iRep = this;
    aRef.iSet = &(arr[aIndex].iName);
    }


TInt HCR::TRepositoryCompiled::GetValue(const TSettingRef& aRef, UValueWord& aValue)
    {
    HCR_FUNC("TRepositoryCompiled::GetValue");
    if (!IsWordValue(aRef))
        HCR_TRACE_RETURN(KErrArgument);
        
	SSettingC* sptr = (SSettingC*)(aRef.iSet);
    aValue = sptr->iValue.iLit;
	return KErrNone;
    }

TInt HCR::TRepositoryCompiled::GetLargeValue(const TSettingRef& aRef, UValueLarge& aValue)
    {
    HCR_FUNC("TRepositoryCompiled::GetLargeValue");
    if (!IsLargeValue(aRef))
        HCR_TRACE_RETURN(KErrArgument);

	SSettingC* sptr = (SSettingC*)(aRef.iSet);
    aValue = sptr->iValue.iPtr;
    return KErrNone;
    }


// -- METHODS -----------------------------------------------------------------
//
// TRepositoryFile


HCR::TRepository* HCR::TRepositoryFile::New(const SRepositoryFile* aRepos)
    {
    HCR_FUNC("TRepositoryFile::New");

    __NK_ASSERT_ALWAYS(aRepos != 0);
    return new TRepositoryFile(aRepos);
    }

HCR::TRepositoryFile::TRepositoryFile(const SRepositoryFile* aRepos)
 : iRepos(aRepos)
    {
    HCR_FUNC("TRepositoryFile");
    }

HCR::TRepositoryFile::~TRepositoryFile()
    {
    HCR_FUNC("~TRepositoryFile"); 
	
#ifdef __WINS__
	// On target hardware the iRepos pointer always points to a file in the Core
	// rom image and hence is not memory allocated on kernel heap. Hence it does
	// not need to be freeded. 
	// When running under the emulator the file repositories are loaded into
	// allocated memory which needs to be freed here.

	delete const_cast<SRepositoryFile*>(iRepos);
	iRepos = 0;
#endif // __WINS__

    }
    
TInt HCR::TRepositoryFile::CheckIntegrity()
    {
    HCR_FUNC("TRepositoryFile::CheckIntegrity");
    
    __NK_ASSERT_ALWAYS(this != 0);   
    __NK_ASSERT_ALWAYS(iRepos != 0);
    
	if ((*((TUint32*)&(iRepos->iHdr)) != 0x66524348) || 
		(iRepos->iHdr.iFormatVersion != 0x0001))
        HCR_TRACEMSG_RETURN("File Repository header describes an unsupported repository type", KErrCorrupt); 
	
    HCR_TRACE2("File repository 0x%x contains %05d entries", iRepos, iRepos->iHdr.iNumSettings);
    
    SSettingF* arr = (SSettingF*) (iRepos+1);
    TSettingId prev(0,0);
    TInt rc=0;
    for (int i=0; i < iRepos->iHdr.iNumSettings; i++, arr++)
    	{
    	__NK_ASSERT_ALWAYS(arr != 0);
    	HCR_TRACE3("Checking entry %05d - (0x%x,0x%x)", i, arr->iName.iId.iCat,  arr->iName.iId.iKey);
    	rc = CompareSSettingIds(prev, arr->iName.iId);
	   	// Check for duplicates that reside next to each other
    	if ((i > 0) && (rc == 0))
    		HCR_TRACE_RETURN (KErrAlreadyExists);
    	// Check that the entries are in ascending order	
    	if (rc != -1)
    		HCR_TRACE_RETURN (KErrCorrupt);
    	prev = arr->iName.iId;
		}
    return KErrNone; 
    }

TInt HCR::TRepositoryFile::FindSetting(const TSettingId& aId, TSettingRef& aSetting)
    {
    HCR_FUNC("TRepositoryFile::FindSetting");
 
    __NK_ASSERT_DEBUG(iRepos != 0);
   
    if (iRepos->iHdr.iNumSettings == 0)
        HCR_TRACE_RETURN(KErrNotFound);
    
    SSettingF* arr = (SSettingF*) (iRepos+1);
    int low = 0;
    int high = iRepos->iHdr.iNumSettings-1;
    int mid;
    int com;
    
    while (low<=high)
        {
        mid = (low+high) >> 1;
        com = CompareSSettingIds(aId, arr[mid].iName.iId);
        if (com < 0)
            high = mid-1;
        else if (com > 0)
            low = mid+1;
        else
            {
            aSetting.iRep = this;
            aSetting.iSet = &((arr[mid]).iName);
            return KErrNone;
            }    
        } 
        
    aSetting.iRep = 0;
	aSetting.iSet = 0; 
    return KErrNotFound;
    }


TInt HCR::TRepositoryFile::FindSetting (const TSettingId& aId,
        TSettingRef& aSetting, TInt32& aPosition, TInt32 aLow, TInt32 aHigh)
    {
    HCR_FUNC("TRepositoryFile::FindSetting within the given range");


    __NK_ASSERT_DEBUG(iRepos != 0);

    if (iRepos->iHdr.iNumSettings == 0)
        HCR_TRACE_RETURN(KErrNotFound);

    SSettingF* arr = (SSettingF*) (iRepos+1);
    TInt32 low = aLow;
    TInt32 high = aHigh;
    TInt32 mid;
    TInt32 com;

    while (low<=high)
        {
        mid = (low+high) >> 1;
        com = CompareSSettingIds(aId, arr[mid].iName.iId);
        if (com < 0)
            high = mid-1;
        else if (com > 0)
            low = mid+1;
        else
            {
            aSetting.iRep = this;
            aSetting.iSet = &((arr[mid]).iName);
            aPosition = mid;
            return KErrNone;
            }    
        } 

    aSetting.iRep = 0;
    aSetting.iSet = 0; 
    aPosition = 0;
    return KErrNotFound;
    }




TInt HCR::TRepositoryFile::GetWordSettings(TInt aNum,   
        SSettingId* aIds[], TInt32* aValues[], TSettingType* aTypes[],
        TInt* aErrors[])
    {
    HCR_FUNC("TRepositoryFile::GetWordSettings");


    __NK_ASSERT_DEBUG(iRepos != 0);
   
    if (iRepos->iHdr.iNumSettings == 0)
        return KErrNotFound;

    TInt err = KErrNone;

    TInt32 rMaxIndex = 0;
    TInt32 rMinIndex = 0;
    TInt32 uFirstIndex = 0;
    TInt32 uLastIndex = 0;
    TInt32 rIndex = 0;
    TInt32 uIndex = 0;

    TSettingRef settingRef(NULL, NULL);
    SSettingF* pSetting = NULL;

    //Find position index within the repository for the first and last setting
    //from user supplied array aIds[]
        uIndex = 0;
        TBool isRedefined = EFalse;
        err = KErrNotFound;
        uFirstIndex = 0;
        while(!isRedefined && uIndex < aNum)
            {
            //Find first setting from user array. The importance here is that we   
            //should get value of first setting index in the repository in rMinIndex.
            //This time the  scope of search is whole repository.
            err = this->FindSetting(*aIds[uIndex],settingRef, rMinIndex, 
                    0, iRepos->iHdr.iNumSettings);
            if(err == KErrNotFound)
                {
                *aErrors[uIndex] = err;
                *aValues[uIndex] = 0;
                //Copy type only if user provided aTypes array
                if(aTypes)
                    *aTypes[uIndex] = ETypeUndefined;
                
                //As FindSetting did not find the element, let's challenge with 
                //the next one from aIds[] array
                uIndex ++;
                continue;
                }
            //fatal error here, nothing to do, just exit and return the error code
            else if(err == KErrNotReady || err != KErrNone)
                {
                return err;
                }
            else // err == KErrNone
                {
                //Get the value and type
                pSetting = (SSettingF*) settingRef.iSet;
                //again copy the type value into the user array if it's provided
                if(aTypes)
                    *aTypes[uIndex] = static_cast<TSettingType>(settingRef.iSet->iType); 
            
                //Check for the found type is this word size? If it's not then 
                //indicate error for this setting
                if(*aTypes[uIndex] > ETypeLinAddr)
                    {
                    *aErrors[uIndex] = KErrArgument;
                    *aValues[uIndex] = 0;
                    }
                else
                    {
                    *aErrors[uIndex] = KErrNone;
                    *aValues[uIndex] = pSetting->iValue.iLit.iInt32;
                    }
                
                //Break the loop by setting the redefined status
                isRedefined = ETrue;
                }
            }
        
        //At this point we should find at least one element, store this index in the  
        //local variable, this is used later in the code. Please be noticed we've  
        //also got rMinIndex - first setting index in the repository. 
        if(err == KErrNone)
            uFirstIndex = uIndex;
        else
            //if we are hear it means we did not find any user settings at all
            //we can't do any thing and just return KErrNotFound to indicate
            //this fact
            return KErrNotFound;

        
        
        //Now lets find the last setting
        uIndex = aNum - 1;
        isRedefined = EFalse;
        err = KErrNotFound;
        
        while(!isRedefined && uIndex > uFirstIndex)
            {
            //Find the last setting from user array. The importance here is that we   
            //should get value of first setting index in the repository in 
            //rMinIndex. This time the  scope of search is whole repository.
            err = this->FindSetting(*aIds[uIndex],settingRef, rMaxIndex, 
                    rMinIndex, iRepos->iHdr.iNumSettings);
            if(err == KErrNotFound)
                {
                *aErrors[uIndex] = err;
                *aValues[uIndex] = 0;
                if(aTypes)
                    *aTypes[uIndex] = ETypeUndefined;
                
                //As FindSetting did not find the element, let's challenge with 
                //previous one
                uIndex --;
                continue;
                }
            //fatal error here, nothing to do, just exit and return the error code
            else if(err == KErrNotReady || err != KErrNone)
                {
                return err;
                }
            else //err == KErrNone
                {
                pSetting = (SSettingF*) settingRef.iSet;
                if(aTypes)
                    *aTypes[uIndex] = static_cast<TSettingType>(settingRef.iSet->iType); 

                //Check for the found type is this word size? If it's not then indicate
                //error for this setting
                if(*aTypes[uIndex] > ETypeLinAddr)
                    {
                    *aErrors[uIndex] = KErrArgument;
                    *aValues[uIndex] = 0;
                    }
                else
                    {
                    *aErrors[uIndex] = KErrNone;
                    *aValues[uIndex] = pSetting->iValue.iLit.iInt32;
                    }
                
                isRedefined = ETrue;
                }
            }

        //At this point we found the last setting, store it's user array index in   
        //the local variable, this is used later in the code. Please be noticed   
        //we've also got rMaxIndex - last setting index in the repository.
        if(err == KErrNone)
            uLastIndex = uIndex;
        else
            //if we are here we did not find any other elements than was found
            //in previous iteration then just stop here
            return KErrNotFound;  
        
        //The scope of user array settings in the repository is found. 
        //Let's find all other settings from user array. Bare in mind the low
        //bound for the repository index is increased each iteration to optimize the
        //search time.
        for(uIndex = uFirstIndex + 1; uIndex < uLastIndex; uIndex ++)
            {
            err = this->FindSetting(*aIds[uIndex],settingRef, rIndex, 
                    rMinIndex, rMaxIndex);
            if(err == KErrNotFound)
                {
                *aErrors[uIndex] = err;
                *aValues[uIndex] = 0;
                if(aTypes)
                    *aTypes[uIndex] = ETypeUndefined;

                //As FindSetting did not find the element, let's challenge with 
                //another one
                continue;
                }
            else if(err == KErrNotReady || err != KErrNone)
                {
                return err;
                }
            else //err == KErrNone
                {

                pSetting = (SSettingF*) settingRef.iSet;
                
                TSettingType type = static_cast<TSettingType>(settingRef.iSet->iType); 
                if(aTypes != NULL)
                    *aTypes[uIndex] = type; 

                //Check for the found type is this word size? If it's not then indicate
                //error for this setting
                if(type > ETypeLinAddr)
                    {
                    *aErrors[uIndex] = KErrArgument;
                    *aValues[uIndex] = 0;
                    }
                else
                    {
                    *aErrors[uIndex] = KErrNone;
                    *aValues[uIndex] = pSetting->iValue.iLit.iInt32;
                    }

                rMinIndex = rIndex + 1;
                }

            }

        return KErrNone;
    }



void HCR::TRepositoryFile::GetSettingRef(TInt32 aIndex, 
         HCR::TSettingRef& aSetRef)
    {
    __NK_ASSERT_DEBUG(iRepos != 0);
    __NK_ASSERT_DEBUG(aIndex >= 0 && aIndex < iRepos->iHdr.iNumSettings);

    if (iRepos->iHdr.iNumSettings == 0)
        {
        aSetRef.iRep = NULL;
        aSetRef.iSet = NULL;
        }

    SSettingF* arr = (SSettingF*)(iRepos + 1);
    
    aSetRef.iRep = this;
    aSetRef.iSet = &(arr[aIndex].iName);
    }




TInt HCR::TRepositoryFile::FindNumSettingsInCategory(TCategoryUid aCatUid,
        TInt32& aFirst, TInt32& aLast)
    {
    HCR_FUNC("TRepositoryFile::FindNumSettingsInCategory");

    __NK_ASSERT_DEBUG(iRepos != 0);

    if (iRepos->iHdr.iNumSettings == 0)
        HCR_TRACE_RETURN(KErrNotFound);

    SSettingF* arr = (SSettingF*) (iRepos+1);
    TInt32 low = 0;
    TInt32 high = iRepos->iHdr.iNumSettings-1;
    TInt32 mid = 0;
    TInt32 com = 0;


    //Let's find any setting within the category, mid will store the setting 
    //index in the repository
    while (low<=high)
        {
        mid = (low+high) >> 1;
        com = CompareByCategory(aCatUid, arr[mid].iName.iId);
        if (com < 0)
            high = mid-1;
        else if (com > 0)
            low = mid+1;
        else
            {
            break;
            }    
        } 

    // If no one setting with the given category was found the return error  
    // to the user
    if(low > high)
        {
        aFirst = 0;
        aLast  = 0;
        return KErrNotFound;
        }

    //Search the first element within the category
    low = mid;
    while(low > 0 && arr[low].iName.iId.iCat == aCatUid)
        {
        low --;
        }
    //Check the boundary conditions, there are two cases when we exit the loop
    //either we found an element which category is not one we are looking for or
    //we reach the beggining of the repository. If we reach the beggining of the
    //repository we don't really know is it because this is last elment or it
    //has required aCatUid, so we check these two conditions below
    if(low == 0 && arr[low].iName.iId.iCat == aCatUid)
        aFirst = low;
    //We finish the loop either reaching the setting which category id is not
    //what we need or this is first setting in the repository again with another
    //category, so in both case we throw this element from the account.
    else
        aFirst = low + 1;


    //Search the last element within the category
    high = mid;
    while(high < iRepos->iHdr.iNumSettings && arr[high].iName.iId.iCat == aCatUid)
        {
        high ++;
        }
    //Same situation as above, boundary conditions
       if(high == (iRepos->iHdr.iNumSettings - 1) && arr[high].iName.iId.iCat == aCatUid)
           aLast = high;
       else
           aLast = high -1;

    return KErrNone;
    }




TInt HCR::TRepositoryFile::GetValue(const TSettingRef& aRef, UValueWord& aValue)
    {
    HCR_FUNC("TRepositoryFile::GetValue");

    if (!IsWordValue(aRef))
        HCR_TRACE_RETURN(KErrArgument);
        
	SSettingF* sptr = (SSettingF*)(aRef.iSet);
    aValue = sptr->iValue.iLit;
	return KErrNone;
    }


TInt HCR::TRepositoryFile::GetLargeValue(const TSettingRef& aRef, UValueLarge& aValue)
    {
    HCR_FUNC("TRepositoryFile::GetLargeValue");

    if (!IsLargeValue(aRef))
        HCR_TRACE_RETURN(KErrArgument);

	SSettingF* sptr = (SSettingF*)(aRef.iSet);
	TRepositoryFile *rptr = (TRepositoryFile *)(aRef.iRep);
	
    aValue.iData = (TUint8*) rptr->iRepos;
	aValue.iData += rptr->iRepos->iLSDfirstByteOffset+sptr->iValue.iOffset;
	
    return KErrNone;
    }


// -- FUNCTIONS ---------------------------------------------------------------

#ifndef HCRTEST_NO_KEXT_ENTRY_POINT
#ifndef __WINS__
DECLARE_EXTENSION_WITH_PRIORITY(KExtensionMaximumPriority)
#else
DECLARE_STANDARD_EXTENSION()
#endif // __WINS__
	{
	HCR_FUNC("InitExtension");

	HCR::MVariant* varPtr = CreateHCRVariant();
	if (varPtr==0)
		HCR_TRACE_RETURN(KErrNoMemory);

	//Call of the "placement" new operator, which constructs the HCR object on 
	//the global memory address defined by gHCR and initialized with the same
	//data given by constructor below
	new(&gHCR) HCR::HCRInternal(varPtr);

	TInt err = HCRSingleton->Initialise();

	if (err != KErrNone)
		HCR_TRACE_RETURN(err);

	return err;
	}
#endif // HCRTEST_NO_KEXT_ENTRY_POINT

// -- Implementation of local functions
#ifndef __WINS__
TInt SearchEntryInTRomDir(const TRomDir* aActDir, const TPtrC aFileName, TRomEntry* &aEntry)
	{
	HCR_FUNC("SearchEntryInTRomDir");
	TInt retVal = KErrNotFound;
	HCR_TRACE2("--- aFileName: %S (%d)", &aFileName, aFileName.Length());
	
	if( aActDir == 0)
		{
		HCR_TRACE_RETURN(retVal);
		}
	
	TInt dirSize = aActDir->iSize;
	aEntry = (TRomEntry*)&aActDir->iEntry;
	HCR_TRACE3("--- dirSize: 0x%08x (%d), aEntry: 0x%08x", dirSize, dirSize, aEntry);
	
	TBool found = EFalse;
	while( !found )
		{
		TInt nameLength = (aEntry->iNameLength)<<1;
		
		// Uncommnet to get dump of ROM data when debugging....
		// HCR_TRACE0("Begin of loop...");
		// HCR_HEX_DUMP_ABS((TUint8 *)aEntry, sizeof(TRomEntry)+(nameLength - 2) );
		const TText* entryName = &aEntry->iName[0];
		HCR_TRACE1("--- entryName length: %d", nameLength);
		TBuf<512> newEntryName( nameLength);
		for( TInt i = 0; i != nameLength; ++i)
			{
			newEntryName[i] = (unsigned char)('A' <= entryName[i] && 'Z' >= entryName[i]? entryName[i]+('a'-'A'): entryName[i]);
			}		
				
		HCR_TRACE6("--- aFileName: %S (%d/%d), newEntryName: %S (%d/%d)", &aFileName, aFileName.Length(), aFileName.Size(), &newEntryName, newEntryName.Length(), newEntryName.Size());
		TInt r = aFileName.Compare(newEntryName);
		HCR_TRACE1("--- result of CompareFileNames: 0x%08x", r);
		
		if ( r == 0)
			{
			found = ETrue;
			HCR_TRACE1("--- aEntry: 0x%08x", aEntry);
			}
		else
			{
		
			TInt entrySize = sizeof(TRomEntry) + (nameLength - 2);
			HCR_TRACE2("--- entrySize: 0x%08x, (%d)", entrySize, entrySize);
			
			// The entrySize must be aligned to 4 bytes boundary
			entrySize = ((entrySize&0x03) == 0 ? entrySize : ((entrySize&0xfffffffc) + 4));
			HCR_TRACE2("--- entrySize: 0x%08x, (%d)", entrySize, entrySize);
			
			aEntry = (TRomEntry*)((char *)aEntry + entrySize);
			dirSize -= entrySize;
			HCR_TRACE2("--- aEntry: 0x%08x, dirSize:%d", aEntry, dirSize);
			if( dirSize <= 0)
				{
				break;
				}
			}
		}
		
	if( found)
		{
		retVal = KErrNone;
		}
		
	HCR_TRACE_RETURN(retVal);		
	}
	
#endif // !__WINS__


TInt SearchCoreImgRepository(HCR::TRepository*& aRepos, const TText * aFileName)
    {
    HCR_FUNC("SearchCoreImgRepository(TRepository*& aRepos, TText & aFileName)");
    
    TInt retVal = KErrNotFound;

	// Convert aFileName to directory entry style Unicode 
	const TText* p = aFileName;
	
	if( *p == 0)
		{
		// Empty file name -> return with KErrNotFound!
		HCR_TRACE_RETURN(retVal);
		}
		
	while( *(++p)) {};					// Search the end of file name string.
	TInt nameLen=(TInt)(p-aFileName);	
	
	HCR_TRACE2("--- aFileName: %s (%d)", aFileName, nameLen );
	
	TBuf<256> origFileName;
	origFileName.Append((const TText*)aFileName, nameLen);
	HCR_TRACE2("--- origFileName: %S (%d)", &origFileName, origFileName.Length());
	

#ifdef __WINS__
    TBuf<KMaxFileName> wholeFilePath; 
	void* reposBuf = 0;
    
#ifdef __VC32__
	
#ifdef _DEBUG
	// - wins udeb version
    wholeFilePath.Copy((const TText*)"\\EPOC32\\RELEASE\\WINS\\UDEB\\");
#else
    // - wins urel version
    wholeFilePath.Copy((const TText*)"\\EPOC32\\RELEASE\\WINS\\UREL\\");
#endif

#else

#ifdef _DEBUG
    // - winscw udeb version
    wholeFilePath.Copy((const TText*)"\\EPOC32\\RELEASE\\WINSCW\\UDEB\\");
#else
    // - winscw urel version
    wholeFilePath.Copy((const TText*)"\\EPOC32\\RELEASE\\WINSCW\\UREL\\");
#endif
    
#endif    
    
    for( TInt j = 0; j < nameLen; ++j)
          {
          wholeFilePath.Append( origFileName[j] );
          }
    
    HCR_TRACE3("--- epoc emulator file path: %S (%d/%d)", &wholeFilePath, wholeFilePath.Length(), wholeFilePath.Size());
    
    TInt length = wholeFilePath.Length();
    
    NKern::ThreadEnterCS();
    TCHAR* chFilePath = new TCHAR[length+1];
    NKern::ThreadLeaveCS();
    
    for(int loop=0;loop<length;++loop) 
        {
        chFilePath[loop] = wholeFilePath[loop];
        }
    chFilePath[length] = '\0';
    
    //try to locate file
    WIN32_FIND_DATAW wfd;
    HANDLE hFile = FindFirstFile(chFilePath, &wfd);
    TBool foundFile = EFalse;
    if (hFile == INVALID_HANDLE_VALUE)
        {
        HCR_TRACE0("--- file not found in \\sys\\bin; try \\sys\\data");
        
#ifdef __VC32__
    
#ifdef _DEBUG
        // - wins udeb version
        wholeFilePath.Copy((const TText*)"\\EPOC32\\RELEASE\\WINS\\UDEB\\Z\\sys\\data\\");
#else
        // - wins urel version
        wholeFilePath.Copy((const TText*)"\\EPOC32\\RELEASE\\WINS\\UREL\\Z\\sys\\data\\");
#endif

#else

#ifdef _DEBUG
        // - winscw udeb version
        wholeFilePath.Copy((const TText*)"\\EPOC32\\RELEASE\\WINSCW\\UDEB\\Z\\sys\\data\\");
#else
        // - winscw urel version
        wholeFilePath.Copy((const TText*)"\\EPOC32\\RELEASE\\WINSCW\\UREL\\Z\\sys\\data\\");
#endif
    
#endif  
        
        for( TInt i = 0; i < nameLen; ++i)
            {
            wholeFilePath.Append( origFileName[i] );
            }
            
        HCR_TRACE3("--- epoc emulator file path: %S (%d/%d)", &wholeFilePath, wholeFilePath.Length(), wholeFilePath.Size());
            
        length = wholeFilePath.Length();
        
        NKern::ThreadEnterCS();
        delete[] chFilePath;
        chFilePath = new TCHAR[length+1];
        NKern::ThreadLeaveCS();
        
        for(int loop=0;loop<length;++loop) 
            {
            chFilePath[loop] = wholeFilePath[loop];
            }
        chFilePath[length] = '\0';
        
        hFile = FindFirstFile(chFilePath, &wfd);
        
        if (hFile == INVALID_HANDLE_VALUE)
            {
            HCR_TRACE0("--- file not found in \\sys\\data");
            }
        else
            {
            HCR_TRACE0("--- file found in \\sys\\data");
            foundFile = ETrue;        
            }
        }
    else
        {
        HCR_TRACE0("--- file found in \\sys\\bin");
        foundFile = ETrue;
        }
    
    if(!foundFile)
        {
        // No file found; release memory and return
        NKern::ThreadEnterCS();
        delete[] chFilePath;
        NKern::ThreadLeaveCS();
        
        HCR_TRACE_RETURN(KErrNotFound);
        }
        

    __NK_ASSERT_ALWAYS(wfd.nFileSizeHigh==0);
            
    DWORD num_read = 0;    
    retVal = KErrNone;
    
    NKern::ThreadEnterCS();
    reposBuf = new BYTE[wfd.nFileSizeLow];
    NKern::ThreadLeaveCS();
    
    if(reposBuf == NULL)
		{
        HCR_TRACEMSG_RETURN("--- Error allocating memory for reading file", KErrNoMemory);
		}
    else
        {
        hFile = CreateFile(chFilePath, GENERIC_READ,          // open for reading
                FILE_SHARE_READ,       // share for reading
                NULL,                  // default security
                OPEN_EXISTING,         // existing file only
                FILE_ATTRIBUTE_NORMAL, // normal file
                NULL); 
        
        BOOL read = ReadFile(hFile, reposBuf, wfd.nFileSizeLow, &num_read, NULL);
        if(!read) 
            {
            retVal = GetLastError();  
            HCR_TRACE1("--- Error reading file %d", GetLastError());
            }
        }

    CloseHandle(hFile);
    NKern::ThreadEnterCS();
    delete[] chFilePath;
    NKern::ThreadLeaveCS();
    
    NKern::ThreadEnterCS();
    aRepos = HCR::TRepositoryFile::New(reinterpret_cast<const HCR::SRepositoryFile *>(reposBuf)); 
    NKern::ThreadLeaveCS();
    
    if (aRepos == NULL)
        {
        retVal = KErrNoMemory;
        }
    
    HCR_TRACE_RETURN(retVal);
    
#else
	
	TBuf<512> fileNameBuf;
	for( TInt i = 0; i != nameLen; ++i)
		{
		fileNameBuf.Append( 'A' <= origFileName[i] && 'Z' >= origFileName[i]? origFileName[i]+('a'-'A'): origFileName[i]);
		fileNameBuf.Append(TChar(0));
		}

	TPtrC fileName(fileNameBuf);
	HCR_TRACE3("--- fileName: %S (%d/%d)", &fileName, fileName.Length(), fileName.Size());

    // Locate ROM Root directory
    TSuperPage& superpage = Kern::SuperPage();
   	TRomRootDirectoryList* romRootDirAddress = (TRomRootDirectoryList*)superpage.iRootDirList;
 
    HCR_TRACE3("--- Superpage: 0x%08x, ROM root dir list: 0x%08x (Num of root dirs:%d)", &superpage, romRootDirAddress, romRootDirAddress->iNumRootDirs );

	// Search the root directory which is match to the current hardware variant
    TUint hardwareVariant 	   = superpage.iActiveVariant;
    TInt variantIndex;
    TRootDirInfo* rootDirInfo = 0;
    
    for(variantIndex = 0; variantIndex < romRootDirAddress->iNumRootDirs; ++variantIndex )
	    {
	    HCR_TRACE3("--- variantIndex:%d, current hardware variant: 0x%08x, root dir hardware variant:0x%08x", variantIndex, hardwareVariant, romRootDirAddress->iRootDir[variantIndex].iHardwareVariant);
	    
    	if( romRootDirAddress->iRootDir[variantIndex].iHardwareVariant == hardwareVariant)
	    	{
	    	rootDirInfo = &romRootDirAddress->iRootDir[variantIndex]; 
	    	break;
	    	}
	    }
    
    if( rootDirInfo == 0 )
	    {
	    // Not found root directory for this hardware variant
	    HCR_TRACE_RETURN(retVal);
	    }
    
	TRomDir* romDir = (TRomDir*)rootDirInfo->iAddressLin;

	HCR_TRACE3("--- romDir: 0x%08x (files:0x%08x, entries:0x%08x)", romDir, romDir->FileCount(), romDir->EntryCount() );
	TRomEntry* entry = (TRomEntry*)&romDir->iEntry;
	
	// We are searching in \sys\bin\ and \sys\Data\ directory only	
	TPtrC level1DirName((const TText*)"s\0y\0s\0", 6);		// Unicode, because the entry names are unicode too.
	TPtrC level2Dir1Name((const TText*)"b\0i\0n\0", 6);
	TPtrC level2Dir2Name((const TText*)"d\0a\0t\0a\0", 8);		// Originally \sys\Data however we search all entry in lower case
	
	TInt r = SearchEntryInTRomDir(romDir, level1DirName, entry);
	HCR_TRACE1("--- result of SearchEntryInTRomDir: 0x%08x", r);

	if( r == KErrNone)
		{
		// \sys directory found.
		romDir = (TRomDir*)entry->iAddressLin;
		HCR_TRACE1("--- romDir: 0x%08x ", romDir);
		
		TRomDir* parentDir = romDir;
		// Search in \sys\bin directory
		r = SearchEntryInTRomDir(romDir, level2Dir1Name, entry);
	
		HCR_TRACE1("--- result of SearchEntryInTRomDir: 0x%08x", r);
		if( r == KErrNone)
			{
			// \sys\bin directory found
			romDir = (TRomDir*)entry->iAddressLin;
			HCR_TRACE1("--- romDir: 0x%08x ", romDir);
			// Search the repository file
			r = SearchEntryInTRomDir(romDir, fileName, entry);
			
			HCR_TRACE1("--- result of SearchEntryInTRomDir: 0x%08x", r);
			if( r == KErrNone)
				{
				// Repository file found
				retVal = KErrNone;				
				HCR_TRACE1("--- Repository address: 0x%08x ", entry->iAddressLin);
#ifdef __EPOC32__
			// HCR design requires the core image file repository to be in the
			// unpaged portion of the core ROM image. This check will Fault the
			// kernel startup if this is not found to be the case, perhaps due 
			// to mis-configured obey files.
			// Skipped on emulator builds as Epoc class in platform.h not
			// defined. Hence support for core images not supported. 
			__NK_ASSERT_ALWAYS(ROMAddressIsInUnpagedSection((TLinAddr)entry->iAddressLin));   
#endif
				NKern::ThreadEnterCS();
				aRepos = HCR::TRepositoryFile::New(reinterpret_cast<const HCR::SRepositoryFile *>(entry->iAddressLin));	
				NKern::ThreadLeaveCS();
				if (aRepos == NULL)
                        retVal = KErrNoMemory;					
                        
                HCR_TRACE_RETURN(retVal);
				}
			}

		// \sys\bin directory or repository file in \sys\bin directory not found.    
		// Search \sys\Data directory
		romDir = parentDir;
		r = SearchEntryInTRomDir(romDir, level2Dir2Name, entry);
		HCR_TRACE1("--- result of SearchEntryInTRomDir: 0x%08x", r);
		if( r == KErrNone)
			{
			// \sys\Data directory found
			romDir = (TRomDir*)entry->iAddressLin;
			HCR_TRACE1("--- romDir: 0x%08x ", romDir);
			
			// Search repository file
			r = SearchEntryInTRomDir(romDir, fileName, entry);
			
			HCR_TRACE1("--- result of SearchEntryInTRomDir: 0x%08x", r);
			if( r == KErrNone)
				{
				// Repository file found    
				retVal = KErrNone;				
				HCR_TRACE1("--- Repository address: 0x%08x ", entry->iAddressLin);
#ifdef __EPOC32__
			// HCR design requires the core image file repository to be in the
			// unpaged portion of the core ROM image. This check will Fault the
			// kernel startup if this is not found to be the case, perhaps due 
			// to mis-configured obey files.
			// Skipped on emulator builds as Epoc class in platform.h not
			// defined. Hence support for core images not supported. 
			__NK_ASSERT_ALWAYS(ROMAddressIsInUnpagedSection((TLinAddr)entry->iAddressLin));   
#endif
				NKern::ThreadEnterCS();
				aRepos = HCR::TRepositoryFile::New(reinterpret_cast<const HCR::SRepositoryFile *>(entry->iAddressLin));
				NKern::ThreadLeaveCS();
				if (aRepos == NULL)
                    retVal = KErrNoMemory;					
				}
			}
		}
	
    HCR_TRACE_RETURN(retVal);
#endif //ifdef __WINS__
    }

TInt LocateCoreImgRepository(HCR::TRepository*& aRepos)
    {
    HCR_FUNC("LocateCoreImgRepository");

#ifdef HCRTEST_COREIMG_DONTUSE_ROMHDR
    
    // Use this testing more on Emulator platform
    // and on hardware when ROM Header is not to be used or not implemented
    
	const TText8* hcrfile = (const TText8*) "hcr.dat";
	TInt retVal = SearchCoreImgRepository(aRepos, hcrfile);
	if (retVal != KErrNone)
		return retVal;
	
#else

	const TRomHeader& romHeader = Epoc::RomHeader(); 	// 0x80000000;
	HCR_TRACE2("--- ROM Header: 0x%08x, HCR file address: 0x%08x", &romHeader, romHeader.iHcrFileAddress);
	
	if(romHeader.iHcrFileAddress != 0)
			{
#ifdef __EPOC32__
			// HCR design requires the core image file repository to be in the
			// unpaged portion of the core ROM image. This check will Fault the
			// kernel startup if this is not found to be the case, perhaps due 
			// to mis-configured obey files.
			// Skipped on emulator builds as Epoc class in platform.h not
			// defined. Hence support for core images not supported. 
			__NK_ASSERT_ALWAYS(ROMAddressIsInUnpagedSection((TLinAddr)romHeader.iHcrFileAddress));   
#endif
			NKern::ThreadEnterCS();
			aRepos = HCR::TRepositoryFile::New(reinterpret_cast<const HCR::SRepositoryFile *>(romHeader.iHcrFileAddress));
			NKern::ThreadLeaveCS();
			if (aRepos == 0)
				return KErrNoMemory;
			}
	else
		return KErrNotFound;
		
#endif // HCRTEST_COREIMG_DONTUSE_ROMHDR


	return KErrNone;
    }