telephonyserverplugins/simtsy/src/CSimPhone.cpp
author srilekhas <srilekhas@symbian.org>
Mon, 23 Aug 2010 13:09:49 +0100
branchRCL_3
changeset 63 f1e62dc8c3f9
parent 15 fc69e1e37771
child 65 630d2f34d719
permissions -rw-r--r--
Merge RCL_3 fixes with latest delivery.

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

/**
 @file
*/

#include <testconfigfileparser.h>
#include "CSimPhone.h"
#include "CSimCall.h"
#include "csimsmsmess.h"
#include "CSimPhBkStore.h"
#include "CSimPhBkUsimStore.h"
#include "CSimONStore.h"
#include "CSimSignalStrength.h"
#include "CSimNetworkStatus.h"
#include "CSimDtmf.h"
#include "CSimIndicator.h"
#include "utils.h"
#include "Simlog.h"
#include "CSimPacketService.h"
#include "CSimSat.h"
#include "CSimBatteryCharger.h"
#include "CSimCallBarring.h"
#include "CSimCallForwarding.h"
#include "CSimCallWaiting.h"
#include "CSimPhoneInitialise.h"
#include "CSimPhoneIMSAuth.h"
#include "CSimTsyMode.h"
#include "csimsmartcardauth.h"
#include "csimsmartcardeap.h"

#include "simtsyglobalproperties.h"
#include "et_struct.h"


#define PDD_NAME _L("ECDRV")				// < Name of the WINS Serial Physical Device Driver
#define LDD_NAME _L("ECOMM")				// < Name of the WINS Serial Logical Device Driver

const TInt KPhBkUSimStoreGranularity=5;		// < The granularity of the phonebook store list array.
const TInt KPhonebookStoreGranularity=2;	// < The granularity of the phonebook store list array.

GLDEF_C void SimPanic(TSimPanic aPanicNumber, TInt aLineNumber)
	{
	LOGPHONE3("SMS.TSY Panic %d Line %d", aPanicNumber, aLineNumber);
	(void) aLineNumber;
	_LIT(panicText,"SIM.TSY");
	User::Panic(panicText,aPanicNumber);
	}

//
// CSimPhone
//
void CSimPhone::CloseObj(TAny* aObj)
/**
*	A utility function for cleaning up the stack.
*
* @param aObj a pointer to the CObject to close
*/
	{
	((CObject*)aObj)->Close();
	}

CSimPhone* CSimPhone::NewL(CPhoneBase*& aPhoneBaseRef)
/**
* Standard two phase constructor.
*
* @return CSimPhone  pointer to the phone object created
* @leave Leaves if no memory or object is not created for any reason
*/
	{
	CSimPhone* phone=new(ELeave) CSimPhone(aPhoneBaseRef);
	TCleanupItem newPhoneClose(CloseObj,phone);
	CleanupStack::PushL(newPhoneClose);
	phone->ConstructL();
	CleanupStack::Pop();
	
	aPhoneBaseRef = phone;
	
	return phone;
	}

CSimPhone::CSimPhone(CPhoneBase*& aPhoneBaseRef)
			: iPhoneBaseRef(aPhoneBaseRef),
			  iAPNServiceOffset(4),
			  iUSIMServiceTable(NULL),
			  iUSIMEnabledServiceTable(NULL),
			  iSIMServiceTable(NULL),
			  iModemDetection(RPhone::EDetectedPresent),
			  iMode(RPhone::EModeIdle), 
			  iNtwkMode(RMobilePhone::ENetworkModeUnknown)
	{
	iPhoneId.iError = KErrNotSupported;
	}
	
/**
* A utility function for reading the service tables and setting them up
*
* @param aMaxSupported size of table supported by etelmm
* @param aLookup config file reference to the table to be looked up
* @param pTable pointer to the table to be constructed for the config file table
* @return void does not return anything
*/	
void CSimPhone::PopulateServiceTableL(	TInt aMaxSupported, 
										const TDesC8& aLookup, 
										RMobilePhone::TMobilePhoneServiceTableV1** aTable)
	{

	RMobilePhone::TMobilePhoneServiceTableV1* pTable = NULL;
	pTable = new(ELeave) RMobilePhone::TMobilePhoneServiceTableV1();
	
	// USIM Supported entry's
	const CTestConfigItem* item=CfgFile()->Item(aLookup);
	if (item)
		{	
		TInt count = CTestConfig::CountElements(item->Value(),KStdDelimiter);
		for (TInt i= 0; i< count; i++)
			{
			TInt entry = 0;			
			TUint8* pTableEntry = &(pTable->iServices1To8);		
				
			TInt ret=CTestConfig::GetElement(item->Value(),KStdDelimiter,0,entry);
			if(ret == KErrNone && entry < (aMaxSupported+1)) // This is the maximum supported services by etelmm.h
				{
					TUint8 basic = 0x01;
					pTableEntry += ((entry-1)/8);
					*pTableEntry |= (basic << ((entry-1)%8));
				}
			}
			*aTable = pTable;	
			LOGPHONE2("Successfully populated %s table in simtsy", &aLookup);	
		}
	else	
		{
		delete(pTable);
		pTable = NULL;
		LOGPHONE2("Failed to populat %s table in simtsy", &aLookup);			
		}
	}

/**
 * A utility function for reading the USIM V8 service tables and setting them up
 *
 * @param aTableV8 pointer to the table to be constructed for the config file table
 * @param aLookup config file reference to the table to be looked up
 * @return void does not return anything
 */	
 void CSimPhone::PopulateServiceTableV8L(const TDesC8& aLookup,RMobilePhone::TMobilePhoneServiceTableV8** aTableV8)
 	{

  	RMobilePhone::TMobilePhoneServiceTableV8* pTable = NULL;
 	pTable = new(ELeave) RMobilePhone::TMobilePhoneServiceTableV8();
 	
 	TInt aMaxSupportedV1 = 56;
 	TInt aMaxSupportedV8 = 72;
 	// USIM Supported entry's for V8
 	
 	const CTestConfigItem* item=CfgFile()->Item(aLookup);
 	if (item)
 		{	
 		TInt count = CTestConfig::CountElements(item->Value(),KStdDelimiter);
 		for (TInt i= 0; i< count; i++)
 			{
 			TInt entry = 0;			
 			TUint8* pTableEntry = &(pTable->iServices57To64);		
 				
 			TInt ret=CTestConfig::GetElement(item->Value(),KStdDelimiter,0,entry);
 			if(ret == KErrNone &&  entry > (aMaxSupportedV1) && entry < (aMaxSupportedV8 +1)) // This is the maximum supported services by etelmm.h
 				{
 					entry -= aMaxSupportedV1;
 					TUint8 basic = 0x01;
 					pTableEntry += ((entry-1)/8);
 					*pTableEntry |= (basic << ((entry-1)%8));
 				}
 			}
 			*aTableV8 = pTable;	
 			LOGPHONE2("Successfully populated %s table in simtsy", &aLookup);	
 		}
 	else	
 		{
 		delete(pTable);
 		pTable = NULL;
 		LOGPHONE2("Failed to populate %s table in simtsy", &aLookup);			
 		}
 	}

 
 
 void CSimPhone::SetTestNumberAndReadConfigurationFileL()
     {
     TInt testNumber;
     User::LeaveIfError(GetTestNumber(testNumber));
     iSectionName.Format(KSectionNameFormat,testNumber);
     
     delete iConfigFile;
     iConfigFile = NULL;
         
     TRAPD(err, iConfigFile = CTestConfig::NewLC(iFs, KConfigFileDir, KConfigFilename); CleanupStack::Pop(iConfigFile));
     if( err == KErrNone && iConfigFile->Section(iSectionName) != NULL )
         {
         iConfigSection = iConfigFile->Section(iSectionName);
         }
    else
        {
        LOGPHONE2("Section for requested test number (%d) not found in the Config File",testNumber);
        delete iConfigFile;
        iConfigFile = NULL;

        User::Leave(KErrNotFound);
        }
     
     User::LeaveIfError(SetTestNumberInUse(testNumber));
     }
 
void CSimPhone::ConstructL()
/**
* 2 Phase Construction (Second phase)
* Loads the WINS Physical and Logical Serial Device Drivers.  C32 loads the non-WINS
* serial drivers, so we don't have to worry about these.
* Creates Voice (CSimVoiceLine) and Data lines (CSimDataLine), CConfigReader objects,
* CSimSysAgent objects, CSimSmsMessaging object, CSimPhoneStore object,
* CSimPhoneBook object, and CSimPacketService object.
*/
	{
#if defined (__WINS__)
	TInt r=User::LoadPhysicalDevice(PDD_NAME);
	r=User::LoadLogicalDevice(LDD_NAME);
#endif

	LOGPHONE1("Starting to Load and Parse the Config File");
	
	User::LeaveIfError(iFs.Connect());
	
	SetTestNumberAndReadConfigurationFileL();
	
	CSimTsyMode::InitL(this);

	iReduceTimers = CSimReduceTimers::NewL();

	InitPhoneStatus();
	iSimPhoneInitialise = CSimPhoneInitialise::NewL(this);

	if (CSimTsyMode::GetMode() != CSimTsyMode::ECdmaV1)
		{
		iVoiceLine = CSimVoiceLine::NewL(this,KVoiceLineName);
		iDataLine = CSimDataLine::NewL(this,KDataLineName);
		iSmsMessaging = CSimSmsMessaging::NewL(this);
		iPhoneSecurity = CSimPhoneSecurity::NewL(this);
		iPhoneUSimApp = CSimPhoneUSimApp::NewL(this);
		iPhoneSmartCardApp = CSimPhoneSmartCardApp::NewL(this);
		iPhoneScAuth = CSimSmartCardAuth::NewL(this);

		iPhBkStores=new(ELeave) CArrayFixFlat<CSimPhBkStore*>(KPhonebookStoreGranularity);
		iPhBkUSimStores=new(ELeave) CArrayFixFlat<CSimPhBkUSimStore*>(KPhBkUSimStoreGranularity);

		FindAndCreatePhBkStoresL();
		CreateONStoreL();

		iSignalStrength=CSimSignalStrength::NewL(this);
		iNetworkStatus=CSimNetworkStatus::NewL(this);
		iDtmf=CSimDtmf::NewL(this);

		}
		
	const TInt capacityServiceTable =  56;
	PopulateServiceTableL(capacityServiceTable,KUSIMServiceTableSupportedEntrys(),	&iUSIMServiceTable);
	PopulateServiceTableL(capacityServiceTable,USIMServiceTableEnabledEntrys(), &iUSIMEnabledServiceTable);
	PopulateServiceTableL(capacityServiceTable,KSIMServiceTable(),	&iSIMServiceTable);
		
	/* Updating the V8 features */
	PopulateServiceTableV8L(KUSIMServiceTableSupportedEntrys(),&iUSIMServiceTableV8);
	
	if (ServiceEnabled(iUSIMServiceTable,iAPNServiceOffset,RMobilePhone::KUstACL))
		{
		TInt entries = 0;
		const CTestConfigItem* aPNListItem=CfgFile()->Item(KAccessPointNameList);
		if (aPNListItem)
			{
			TInt ret=CTestConfig::GetElement(aPNListItem->Value(),KStdDelimiter,0,entries);
			if(ret == KErrNone && entries > 0)
				{
				for (TInt i= 0;i< entries;i++)
					{
					RMobilePhone::TAPNEntryV3 entry;
					entry.iApn.Zero();
					TPtrC8 startupAPNList;
					ret=CTestConfig::GetElement(aPNListItem->Value(),KStdDelimiter,i+1,startupAPNList);
					entry.iApn.Copy(startupAPNList);
					iAPNList.Append(entry);	
					}
				}
			}			
		}

	TPtrC8 IMSI;
	iSubscriberId.iError = KErrNone;
	const CTestConfigItem* itemSubscriberId=CfgFile()->Item(KSubscriberId);
    if( itemSubscriberId != NULL )
        {
        TInt ret=CTestConfig::GetElement(itemSubscriberId->Value(),KStdDelimiter,0,IMSI);
        if( ret != KErrNone )
			{
            LOGPARSERR("IMSI",ret,0,&KSubscriberId);
			}
        // coverity[check_return]
        CTestConfig::GetElement(itemSubscriberId->Value(),KStdDelimiter,1,iSubscriberId.iError);
        iSubscriberId.iIMSI.Copy(IMSI);
        }
	else
		{
		iSubscriberId.iIMSI.Copy(KSubscriberIdDefault);
		}

	iNtwkMode=(RMobilePhone::TMobilePhoneNetworkMode)CfgFile()->ItemValue(KNetworkMode,KNetworkModeDefault);
	//get phone id from config file
	GetPhoneIdAndCaps();

	LOGPHONE1("Loading the version configuration");
	iSimTsyVersion = CfgFile()->ItemValue(KTsyVersionNumber,KSimTsyDefaultVersion);
	LOGPHONE2("Simulated SIMTSY version: %d", iSimTsyVersion);
	
	iPacketService = CSimPacketService::NewL(this);
	iBatteryCharger = CSimBatteryCharger::NewL(this);
	iSat = CSimSat::NewL(this);
	iIndicator = CSimIndicator::NewL(this);
	iPhoneIMSAuth = CSimPhoneIMSAuth::NewL(this);

	iCallBarring=CSimCallBarring::NewL(this);
	iCallForwarding=CSimCallForwarding::NewL(this);
	iCallWaiting=CSimCallWaiting::NewL(this);
	iUsimRelease6=CSimUsimR6::NewL(this);

	iNetworkModeArray = new (ELeave) CArrayFixFlat<TNetworkModeBundle>(KGranularity);

	iPhoneScEapMan = CSimSmartCardEapManager::NewL(this);

	iNetworkModeTimer = CSimTimer::NewL(this);
	//< Read in all network mode data
	TInt count = CfgFile()->ItemCount(KNetworkMode);
	const CTestConfigItem* item = NULL;
	for (TInt i = 0; i < count; ++i)
		{
		item = CfgFile()->Item(KNetworkMode,i);
		if (!item)
			break;
		
		TInt duration(0); 
		TInt networkMode(0);
		TInt ret = CTestConfig::GetElement(item->Value(),KStdDelimiter,0,duration);
		if (ret != KErrNone)
			break;
		ret = CTestConfig::GetElement(item->Value(),KStdDelimiter,1,networkMode);
		if (ret != KErrNone)
			break;
		RMobilePhone::TMobilePhoneNetworkMode mode;
		if(networkMode == 1)
			{
			mode = RMobilePhone::ENetworkModeUnregistered;
			}
		else if(networkMode == 2)
			{
			mode = RMobilePhone::ENetworkModeGsm;
			}
		else if(networkMode == 3)
			{
			mode = RMobilePhone::ENetworkModeAmps;
			}
		else if(networkMode == 6)
			{
			mode = RMobilePhone::ENetworkModeWcdma;
			}
		else if(networkMode == 7)
			{
			mode = RMobilePhone::ENetworkModeTdcdma;
			}
		else
			{
			mode = RMobilePhone::ENetworkModeUnknown;
			}
		TNetworkModeBundle networkModeData;
		networkModeData.iDuration = duration;
		networkModeData.iNetworkMode = mode;
		
		TRAP_IGNORE(iNetworkModeArray->AppendL(networkModeData));
		}
	
	if (iNetworkModeArray->Count() != 0)
		{
		iNetworkModeIndex = 0;
		TNetworkModeBundle initialNetworkMode = iNetworkModeArray->At(0);
		iNtwkMode = initialNetworkMode.iNetworkMode;		// set to new value if NetworkMode present in config.txt file
		iTimerCallBackNetworkMode.SetHandle(this);
		iNetworkModeTimer->Start(initialNetworkMode.iDuration, &iTimerCallBackNetworkMode);
		}
	// end of network mode simulation setup
	iTestNumberObserver = CSimTestNumberObserver::NewL(*this);
	
	LOGPHONE1("Completed Loading and Parsing the Config File");
	}

CSimReduceTimers* CSimPhone::GetReduceTimersSubject()
	{
	return iReduceTimers;
	}

TInt CSimPhone::GetTestNumber(TInt& aTestNumber)
/**
 * Retrieve the test number.
 */
	{
	TInt  ret = RProperty::Get(KUidPSSimTsyCategory, KPSSimTsyTestNumber, aTestNumber);

	if (ret == KErrNone  &&  aTestNumber >= 0)
		{
		LOGPHONE2("Got system property KUidPSSimTsyCategory/KPSSimTsyTestNumber. testNumber=%d", aTestNumber);
		}
	else
		{
		aTestNumber = KDefaultTestNumber;
		LOGPHONE2("Using the default test number. testNumber=%d", aTestNumber);
		}

	return KErrNone;
	}


TInt CSimPhone::SetTestNumberInUse(TInt aTestNumber)
/**
 * Set the test number that is actually used by the TSY
 */
	{
	TInt  ret = RProperty::Set(KUidPSSimTsyCategory, KPSSimTsyTestNumberInUse, aTestNumber);

	if (ret == KErrNone)
		{
		LOGPHONE2("Set system property KUidPSSimTsyCategory/KPSSimTsyTestNumberInUse. testNumber=%d", aTestNumber);
		}

	return ret;
	}


CSimPhone::~CSimPhone()
/**
* Standard destructor.
* Any objects created by the ::ConstructL() function should be destroyed here.
* The objects to be destroyed are:
* CSimVoiceLine, CSimDataLine, CConfigReader, CSimSysAgent, CSimSmsMessaging
* CSimPhoneStore, CSimPhoneBook, CSimPacketService
*/
	{
	iPhoneBaseRef = NULL;

	delete iNetworkModeTimer;

	delete iDtmf;
	delete iNetworkStatus;
	delete iSignalStrength;
	delete iIndicator;
	delete iBatteryCharger;
	delete iPhoneIMSAuth;
	delete iPhoneScEapMan;
	delete iPhoneScAuth;
	delete iUsimRelease6;
	
	iFs.Close();
	if(iDataLine)
		iDataLine->Close();
	if(iVoiceLine)
		iVoiceLine->Close();
	if(iSmsMessaging)
		iSmsMessaging->Close();
	if(iPhoneSecurity)
		iPhoneSecurity->Close();
	if(iPhoneUSimApp)
		iPhoneUSimApp->Close();
	if(iPhoneSmartCardApp)
		iPhoneSmartCardApp->Close();
			
	if (iUSIMServiceTable != NULL)		
		delete(iUSIMServiceTable);
	if (iUSIMEnabledServiceTable != NULL)		
		delete(iUSIMEnabledServiceTable);	
	if (iSIMServiceTable != NULL)		
		delete(iSIMServiceTable);
	if (iUSIMServiceTableV8 != NULL)		
		delete(iUSIMServiceTableV8);
	
	if( iPhBkStores != NULL )
		{
		TInt storeCount=iPhBkStores->Count();
		for(TInt i=0;i<storeCount;i++)
			{
			iPhBkStores->At(i)->Close();
			}
		delete iPhBkStores;
		}
	if( iPhBkUSimStores != NULL )
	    {
		TInt storeCount=iPhBkUSimStores->Count();
		for(TInt i=0;i<storeCount;i++)
			{
			iPhBkUSimStores->At(i)->Close();
			}
		delete iPhBkUSimStores;
		}
	
	if(iONStore)
		{
		iONStore->Close();
		iONStore = NULL;
		}
	if( iNetworkModeArray != NULL )
		{
		iNetworkModeArray->Delete(0,iNetworkModeArray->Count());
		delete iNetworkModeArray;
		}
	if( iPacketService!= NULL )
	    {
		iPacketService->Close();
	    }
	if( iSat != NULL )
	    {
		iSat->Close();
	    }
	if( iConfigFile == NULL)
	    {
	    // In this case if iConfigSection exists then it is not owned
	    // by iConfigFile - this is cos there was no config file for
	    // this SIM TSY.
	    delete iConfigSection;
	    }
	delete iConfigFile;
    delete iCallBarring;
    delete iCallForwarding;
    delete iCallWaiting;

	if (iSetCallProcessingSuspendStateTimer)
		{
		iSetCallProcessingSuspendStateTimer->Cancel();
		delete iSetCallProcessingSuspendStateTimer;		
		}
	delete iSimPhoneInitialise;
	
	delete iReduceTimers;
	delete iTestNumberObserver;
	
	CSimTsyMode::FreeMode();
	LOGPHONE1("CSimPhone Destroyed");
	}

void CSimPhone::InitPhoneStatus()
	{
	const CTestConfigItem* item = NULL;
	TInt ret = KErrNone;

	LOGPHONE1("Starting to Load and Parse Phone Config parameters");

	item = CfgFile()->Item(KPhoneStatus);
	if(!item)
		return;

	TInt modemDetection, mode;
	ret = CTestConfig::GetElement(item->Value(), KStdDelimiter, 0, modemDetection);
	if (ret == KErrNone)
		{
		iModemDetection = (RPhone::TModemDetection)modemDetection;
		}
	ret=CTestConfig::GetElement(item->Value(), KStdDelimiter, 1, mode);
	if (ret != KErrNone)
		{
		iMode = (RPhone::TMode)mode;
		}
	}

void CSimPhone::FindAndCreatePhBkStoresL()
	{
	TInt count=CfgFile()->ItemCount(KPhBkStore);
	const CTestConfigItem* item=NULL;
	TInt ret=KErrNone;

	LOGPHONE1("Starting to Load and Parse PhoneBookStore Config parameters");
	TInt i;
	for(i=0;i<count;i++)
		{
		item=CfgFile()->Item(KPhBkStore,i);
		if(!item)
			break;

		TPtrC8 phonebookName;
		TInt maxNumSlots, telNumMaxLen, alphaTagMaxLen;
		ret=CTestConfig::GetElement(item->Value(),KStdDelimiter,0,phonebookName);
		if(ret!=KErrNone)
			{
			LOGPARSERR("phonebookName",ret,0,&KPhBkStore);
			continue;
			}
		ret=CTestConfig::GetElement(item->Value(),KStdDelimiter,1,maxNumSlots);
		if(ret!=KErrNone)
			{
			LOGPARSERR("maxNumSlots",ret,1,&KPhBkStore);
			continue;
			}

		ret=CTestConfig::GetElement(item->Value(),KStdDelimiter,2,telNumMaxLen);
		if(ret!=KErrNone)
			{
			LOGPARSERR("telNumMaxLen",ret,2,&KPhBkStore);
			continue;
			}

		ret=CTestConfig::GetElement(item->Value(),KStdDelimiter,3,alphaTagMaxLen);
		if(ret!=KErrNone)
			{
			LOGPARSERR("alphaTagMaxLen",ret,3,&KPhBkStore);
			continue;
			}


		CSimPhBkStore* phBkStore=CSimPhBkStore::NewL(this,phonebookName,maxNumSlots,telNumMaxLen,alphaTagMaxLen);
		TCleanupItem newObjClose(CloseObj,phBkStore);
		CleanupStack::PushL(newObjClose);
		iPhBkStores->AppendL(phBkStore);
		CleanupStack::Pop();
		}


	count=CfgFile()->ItemCount(KPhBkUSimStore);

	LOGPHONE1("Starting to Load and Parse USim PhoneBookStore Config parameters");

	for(i=0;i<count;i++)
		{
		item=CfgFile()->Item(KPhBkUSimStore,i);
		if(!item)
			break;

		TPtrC8 phonebookName, phonebookStore;
		TInt maxNumSlots, telNumMaxLen, alphaTagMaxLen;
		TInt additional, maxEmail, maxAdditionalTelNumLen, maxAdditionalTextLen;

		ret=CTestConfig::GetElement(item->Value(),KStdDelimiter,0,phonebookStore);
		if(ret!=KErrNone)
			{
			LOGPARSERR("phonebookStore",ret,0,&KPhBkUSimStore);
			continue;
			}

		ret=CTestConfig::GetElement(item->Value(),KStdDelimiter,1,phonebookName);
		if(ret!=KErrNone)
			{
			LOGPARSERR("phonebookName",ret,1,&KPhBkUSimStore);
			continue;
			}

		ret=CTestConfig::GetElement(item->Value(),KStdDelimiter,2,maxNumSlots);
		if(ret!=KErrNone)
			{
			LOGPARSERR("maxNumSlots",ret,2,&KPhBkUSimStore);
			continue;
			}

		ret=CTestConfig::GetElement(item->Value(),KStdDelimiter,3,telNumMaxLen);
		if(ret!=KErrNone)
			{
			LOGPARSERR("telNumMaxLen",ret,3,&KPhBkUSimStore);
			continue;
			}

		ret=CTestConfig::GetElement(item->Value(),KStdDelimiter,4,alphaTagMaxLen);
		if(ret!=KErrNone)
			{
			LOGPARSERR("alphaTagMaxLen",ret,4,&KPhBkUSimStore);
			continue;
			}

		ret=CTestConfig::GetElement(item->Value(),KStdDelimiter,5,additional);
		if(ret!=KErrNone)
			{
			LOGPARSERR("additional",ret,5,&KPhBkUSimStore);
			continue;
			}

		ret=CTestConfig::GetElement(item->Value(),KStdDelimiter,6,maxEmail);
		if(ret!=KErrNone)
			{
			LOGPARSERR("maxEmail",ret,6,&KPhBkUSimStore);
			continue;
			}

		ret=CTestConfig::GetElement(item->Value(),KStdDelimiter,7,maxAdditionalTelNumLen);
		if(ret!=KErrNone)
			{
			LOGPHONE1("WARNING maxAdditionalTelNumLen missing, defaulting to telNumMaxLen");
			maxAdditionalTelNumLen = telNumMaxLen;
			}
	
		ret=CTestConfig::GetElement(item->Value(),KStdDelimiter,8,maxAdditionalTextLen);
		if(ret!=KErrNone)
			{
			LOGPHONE1("WARNING maxAdditionalTextLen missing, defaulting to alphaTagMaxLen");
			maxAdditionalTextLen = alphaTagMaxLen;
			}

		TInt countStores = iPhBkUSimStores->Count();
		TBool found=EFalse;

		if(phonebookName.Compare(KGsmPhoneBook) == KErrNone)
		{
		TInt countShort = iPhBkStores->Count();
		TBool foundShort=EFalse;

			for(TInt j=0; i < countShort; j++)
			{
				CSimPhBkStore* phBkStore = iPhBkStores->At(j);
				if((phBkStore->Name().Compare(phonebookName) == KErrNone))
					{
					foundShort=ETrue;
					break;
					}
			}
			if(foundShort) //warn but not fail!
				{
				LOGPHONE1("WARNING! PhBkUSimStore and PhBkStore duplicate phonebook configuration");
				}
		}
		for(TInt j=0; j < countStores; j++)
			{
				CSimPhBkUSimStore* phBkStore = iPhBkUSimStores->At(j);
				if((phBkStore->Name().Compare(phonebookName) == KErrNone)&&
					(phBkStore->PhBkStore().Compare(phonebookStore) == KErrNone))
					{
					found=ETrue;
					break;
					}
			}

		if(found)    //This store already exists!!!
			continue;
		
		CSimPhBkUSimStore* phBkStore=CSimPhBkUSimStore::NewL(this,phonebookStore,phonebookName,maxNumSlots,telNumMaxLen,alphaTagMaxLen,additional,maxEmail,maxAdditionalTelNumLen,maxAdditionalTextLen);
		TCleanupItem newObjClose(CloseObj,phBkStore);
		CleanupStack::PushL(newObjClose);
		iPhBkUSimStores->AppendL(phBkStore);
		CleanupStack::Pop();
		}


// Populate the Phonebook Stores after construction, so as not to disturb the config file's
// search pointer
	for(i=0;i<iPhBkStores->Count();i++)
		{
		iPhBkStores->At(i)->PopulateStoreFromConfigFileL();
		}
	for(i=0;i<iPhBkUSimStores->Count();i++)
		{
		iPhBkUSimStores->At(i)->PopulateStoreFromConfigFileL();
		}
	LOGPHONE1("Finished parsing PhBkStores config parameters");
	}

void CSimPhone::CreateONStoreL()
	{
	const CTestConfigItem* item=NULL;
	TInt ret=KErrNone;

	LOGPHONE1("Starting to Load and Parse Own Number Config parameters");

	TInt i=0;
	item=CfgFile()->Item(KONStore,i);
	if(!item)
		return;

	TInt maxNumSlots, telNumMaxLen, alphaTagMaxLen;
	ret=CTestConfig::GetElement(item->Value(),KStdDelimiter,0,maxNumSlots);
	if(ret!=KErrNone)
		{
		LOGPARSERR("maxNumSlots",ret,1,&KONStore);
		return;
		}

	ret=CTestConfig::GetElement(item->Value(),KStdDelimiter,1,telNumMaxLen);
	if(ret!=KErrNone)
		{
		LOGPARSERR("telNumMaxLen",ret,2,&KONStore);
		return;
		}

	ret=CTestConfig::GetElement(item->Value(),KStdDelimiter,2,alphaTagMaxLen);
	if(ret!=KErrNone)
		{
		LOGPARSERR("alphaTagMaxLen",ret,3,&KONStore);
		return;
		}

	iONStore=CSimONStore::NewL(this,maxNumSlots,telNumMaxLen,alphaTagMaxLen);

// Populate the Own Number Store 
	iONStore->PopulateStoreFromConfigFileL();

	LOGPHONE1("Finished parsing Own Number Store config parameters");
	}

TInt CSimPhone::ExtFunc(const TTsyReqHandle aReqHandle, const TInt aIpc, const TDataPackage& aPckg)
/**
* ExtFunc is called by the server when it has a "extended", i.e. non-core ETel request
* for the TSY to process.
* A request handle, request type and request data are passed to the TSY.
*/
	{
	TAny* dataPtr = aPckg.Ptr1();

	TAny* dataPtr2 = aPckg.Ptr2();

	TInt functionValue;

	switch (aIpc)
		{
	case EMobilePhoneGetMultimodeCaps:
		return GetMultimodeCaps(aReqHandle, reinterpret_cast<TUint32*>(dataPtr));

	case EMobilePhoneGetIccAccessCaps:
		return GetIccAccessCaps(aReqHandle,
			REINTERPRET_CAST(TUint32*,dataPtr));

	case EMobilePhoneGetPhoneStoreInfo:
		switch(aPckg.Type())
		{
		// Switch between the alternative implementations of GetPhoneStoreInfo
		case TDataPackage::EPackage1n2u:
			return GetPhoneStoreInfo(aReqHandle,
					aPckg.Des1n(),aPckg.Des2u());
		case TDataPackage::EPackage1n2n:
			return GetPhoneStoreInfo(aReqHandle, aPckg.Des2n(),
					REINTERPRET_CAST(RMobilePhone::TPhoneStoreNameAndIccType*, dataPtr));
		default:
			return NULL;
		}


//		return GetPhoneStoreInfo(aReqHandle,aPckg.Des1n(),aPckg.Des2u());

// Signal Strength
	case EMobilePhoneGetSignalCaps:
		return iSignalStrength->GetSignalCaps(aReqHandle,aPckg.Des1n());

	case EMobilePhoneGetSignalStrength:
		return iSignalStrength->GetSignalStrength(aReqHandle,aPckg.Des1n(),aPckg.Des2n());

	case EMobilePhoneNotifySignalStrengthChange:
		return iSignalStrength->NotifySignalStrengthChange(aReqHandle,aPckg.Des1n(),aPckg.Des2n());

// Indicators
	case EMobilePhoneGetIndicator:
		return iIndicator->GetIndicator(aReqHandle,aPckg.Des1n());

	case EMobilePhoneNotifyIndicatorChange:
		return iIndicator->NotifyIndicatorChange(aReqHandle,aPckg.Des1n());

	case EMobilePhoneGetIndicatorCaps:
		return iIndicator->GetIndicatorCaps(aReqHandle,aPckg.Des1n(), aPckg.Des2n());

// Battery Charger
	case EMobilePhoneGetBatteryInfo:
		return iBatteryCharger->GetBatteryInfo(aReqHandle,aPckg.Des1n());

	case EMobilePhoneNotifyBatteryInfoChange:
		return iBatteryCharger->NotifyBatteryInfoChange(aReqHandle,aPckg.Des1n());

	case EMobilePhoneGetBatteryCaps:
		return iBatteryCharger->GetBatteryCaps(aReqHandle,aPckg.Des1n());

// Network Information
	case EMobilePhoneGetNetworkCaps:
		return iNetworkStatus->GetNetworkCaps(aReqHandle,aPckg.Des1n());

	case EMobilePhoneGetCurrentMode:
		return GetCurrentMode(aReqHandle,aPckg.Des1n());


	case EMobilePhoneNotifyModeChange:
		return NotifyModeChange(aReqHandle,reinterpret_cast<RMobilePhone::TMobilePhoneNetworkMode*>(dataPtr)); 


	case EMobilePhoneGetHomeNetwork:
		return iNetworkStatus->GetHomeNetwork(aReqHandle,aPckg.Des1n());

	case EMobilePhoneGetCurrentNetwork:
		return iNetworkStatus->GetCurrentNetwork(aReqHandle,aPckg.Des1n(),aPckg.Des2n());

	case EMobilePhoneGetCurrentNetworkNoLocation:
		return iNetworkStatus->GetCurrentNetworkNoLocation(aReqHandle,aPckg.Des1n());

	case EMobilePhoneNotifyCurrentNetworkChange:
		return iNetworkStatus->NotifyCurrentNetworkChange(aReqHandle,aPckg.Des1n(),aPckg.Des2n());

	case EMobilePhoneNotifyCurrentNetworkNoLocationChange:
		return iNetworkStatus->NotifyCurrentNetworkNoLocationChange(aReqHandle,aPckg.Des1n());

	case EMobilePhoneGetNetworkRegistrationStatus:
		return iNetworkStatus->GetNetworkRegistrationStatus(aReqHandle,aPckg.Des1n());

	case EMobilePhoneNotifyNetworkRegistrationStatusChange:
		return iNetworkStatus->NotifyNetworkRegistrationStatusChange(aReqHandle,aPckg.Des1n());
	
	case EMobilePhoneGetServiceProviderName:
		return iNetworkStatus->GetCurrentServiceProvider(aReqHandle,aPckg.Des1n());

	case EMobilePhoneGetNetworkName:
		return iNetworkStatus->GetCurrentNetworkName(aReqHandle,aPckg.Des1n(), aPckg.Des2n());
		
//USIM Release6 Infomration
	case EMobilePhoneGetMailboxNumbers:
		return iUsimRelease6->GetMailboxNumbers(aReqHandle,aPckg.Des1n());
		
	case EMobilePhoneNotifyMailboxNumbersChange:
	 	return iUsimRelease6->NotifyMailboxNumbersChange(aReqHandle,aPckg.Des1n());
	 	
	case EMobilePhoneNotifyMessageWaiting:
		return iUsimRelease6->NotifyMessageWaiting(aReqHandle,REINTERPRET_CAST(TInt*, dataPtr));
		
	case EMobilePhoneGetIccMessageWaitingIndicators:
	 	return iUsimRelease6->GetIccMessageWaitingIndicators(aReqHandle,aPckg.Des1n());
	 	
	case EMobilePhoneSetIccMessageWaitingIndicators:
		return iUsimRelease6->SetIccMessageWaitingIndicators(aReqHandle,aPckg.Des1n());
		
	case EMobilePhoneNotifyIccMessageWaitingIndicatorsChange:
		return iUsimRelease6->NotifyIccMessageWaitingIndicatorsChange(aReqHandle,aPckg.Des1n());
	
	case EMobilePhoneNotifyWlanDataChange:
		return iUsimRelease6->NotifyWlanDataChange(aReqHandle,aPckg.Des1n());
	
	case EMobilePhoneNotifyPreferredWlanSIDListChange:
		return iUsimRelease6->NotifyPreferredWlanSIDListChange(aReqHandle);
		
	case EMobilePhoneGetWlanData:
		return iUsimRelease6->GetWlanData(aReqHandle,aPckg.Des1n());
		
	case EMobilePhoneSetWlanData:
		return iUsimRelease6->SetWlanData(aReqHandle,aPckg.Des1n());
	
	case EMobilePhoneGetPreferredWlanSIDsPhase1:
		return iUsimRelease6->GetPreferredWlanSIDsPhase1(aReqHandle, 
			REINTERPRET_CAST(RMobilePhone::TClientId*, dataPtr), 
			REINTERPRET_CAST(TInt*, dataPtr2));
			
	case EMobilePhoneGetPreferredWlanSIDsPhase2:
		return iUsimRelease6->GetPreferredWlanSIDsPhase2(aReqHandle, 
			REINTERPRET_CAST(RMobilePhone::TClientId*, dataPtr), aPckg.Des2n());		
						
	case EMobilePhoneStorePreferredWlanSIDList:
		return iUsimRelease6->StorePreferredWlanSIDList(aReqHandle, aPckg.Des1n());
			
	case EMobilePhoneNotifyStorePreferredNetworksListChange:
		return iUsimRelease6->NotifyStorePreferredNetworksListChange(aReqHandle);
					
	case EMobilePhoneGetPreferredNetworksPhase1:
		return iUsimRelease6->GetPreferredNetworksPhase1(aReqHandle, 
			REINTERPRET_CAST(RMobilePhone::TClientId*, dataPtr), 
			REINTERPRET_CAST(TInt*, dataPtr2));

	case EMobilePhoneGetPreferredNetworksPhase2:
		return iUsimRelease6->GetPreferredNetworksPhase2(aReqHandle, 
			REINTERPRET_CAST(RMobilePhone::TClientId*, dataPtr), aPckg.Des2n());		
			
	case EMobilePhoneStorePreferredNetworksList:
		return iUsimRelease6->StorePreferredNetworksList(aReqHandle, aPckg.Des1n());
		
	case EMobilePhoneSetGbaBootstrapParams:
		return iUsimRelease6->SetGbaBootstrapParams(aReqHandle, aPckg.Des1n(),
		         reinterpret_cast<RMobilePhone::TAID*>(dataPtr2));
	
	case EMobilePhoneNotifyAuthenticateDataChange:
		return iUsimRelease6->NotifyAuthenticateDataChange(aReqHandle, aPckg.Des1n());
	
	case EMobilePhoneGetAuthenticationParams:
		return iUsimRelease6->GetAuthenticationParams(aReqHandle, aPckg.Des1n(),
				aPckg.Des2n());
		
	case EMobilePhoneAuthenticationListPhase1:
		return iUsimRelease6->GetAuthenticationListPhase1(aReqHandle,
				REINTERPRET_CAST(CRetrieveMobilePhoneAuthenticationIds::TAuthRequestData*, dataPtr),
				REINTERPRET_CAST(TInt*, dataPtr2));
		
	case EMobilePhoneAuthenticationListPhase2:
		return iUsimRelease6->GetAuthenticationListPhase2(aReqHandle, 
				REINTERPRET_CAST(RMobilePhone::TClientId*, dataPtr),
				aPckg.Des2n());

// NITZ Network Information
	case EMobilePhoneGetNITZInfo:
		return iNetworkStatus->GetNITZInfo(aReqHandle,aPckg.Des1n());

	case EMobilePhoneNotifyNITZInfoChange:
		return iNetworkStatus->NotifyNITZInfoChange(aReqHandle,aPckg.Des1n());

// Cell Information
			case EMobilePhoneGetCellInfo:
				return iNetworkStatus->GetCellInfo(aReqHandle,aPckg.Des1n());

			case EMobilePhoneNotifyCellInfoChange:
				return iNetworkStatus->NotifyCellInfoChange(aReqHandle,aPckg.Des1n());

// Get Subscriber Info
	case EMobilePhoneGetSubscriberId:
		return GetSubscriberInfo(aReqHandle,aPckg.Des1n());

// DTMF Tx
	case EMobilePhoneGetDTMFCaps:
		return iDtmf->GetDtmfCaps(aReqHandle,aPckg.Des1n());

	case EMobilePhoneNotifyDTMFCapsChange:
		return iDtmf->NotifyDtmfCapsChange(aReqHandle,aPckg.Des1n());

	case EMobilePhoneSendDTMFTones:
		return iDtmf->SendDTMFTones(aReqHandle,aPckg.Des1u());

	case EMobilePhoneStartDTMFTone:
		return iDtmf->StartDTMFTone(aReqHandle,aPckg.Des1n());

	case EMobilePhoneStopDTMFTone:
		return iDtmf->StopDTMFTone(aReqHandle);

	case EMobilePhoneNotifyStopInDTMFString:
		return iDtmf->NotifyStopInDTMFString(aReqHandle);

	case EMobilePhoneContinueDTMFStringSending:
		return iDtmf->ContinueDtmfStringSending(aReqHandle,aPckg.Des1n());
// Phone ID
	case EMobilePhoneGetIdentityCaps:
		return GetIdentityCaps(aReqHandle,*REINTERPRET_CAST(TUint32*, dataPtr));

	case EMobilePhoneGetPhoneId:
		return GetPhoneId(aReqHandle,REINTERPRET_CAST(RMobilePhone::TMobilePhoneIdentityV1*, dataPtr));

// Security (ICC)

	case EMobilePhoneGetSecurityCaps:
	case EMobilePhoneNotifySecurityCapsChange:
	case EMobilePhoneGetLockInfo:
	case EMobilePhoneNotifyLockInfoChange:
	case EMobilePhoneSetLockSetting:
	case EMobilePhoneChangeSecurityCode:
	case EMobilePhoneNotifySecurityEvent:
	case EMobilePhoneVerifySecurityCode:
	case EMobilePhoneAbortSecurityCode:
	case EMobilePhoneGetSecurityCodeInfo:
	case EMobilePhoneNotifySecurityCodeInfoChange:
		return iPhoneSecurity->ExtFunc(aReqHandle, aIpc, aPckg);

//USim Application Support
	case EMobilePhoneEnumerateUSimApplications:
	case EMobilePhoneSetUSimApplicationStatus:
	case EMobilePhoneGetUSimApplicationsInfo:
	case EMobilePhoneNotifyUSimApplicationsInfoChange:
		if(iPhoneUSimApp->FoundUSimAppTags())
			{
			return iPhoneUSimApp->ExtFunc(aReqHandle, aIpc, aPckg);
			}
		else
			{
			return iPhoneSmartCardApp->ExtFunc(aReqHandle, aIpc, aPckg);
			}
			
//SmartCard Application Support
	case EMobilePhoneEnumerateSmartCardApplications:
	case EMobilePhoneGetSmartCardApplicationInfo:
	case EMobilePhoneSetSmartCardApplicationStatus:
	case EMobilePhoneNotifySmartCardApplicationInfoChange:
	case EMobilePhoneGetScFileInfo:
	case EMobilePhoneReadScFile:
	case EMobilePhoneUpdateScFile:
		if(iPhoneSmartCardApp->FoundScAppTags())
			{
			return iPhoneSmartCardApp->ExtFunc(aReqHandle, aIpc, aPckg);
			}
		else
			{
			ReqCompleted(aReqHandle, KErrNotSupported);
			return KErrNone;
			}

		
//USim/SmartCard Application Supoprt
	case EMobilePhoneGetCurrentActiveUSimApplication:
	case EMobilePhoneGetUSimAppsSelectionMode:
	case EMobilePhoneSetUSimAppsSelectionMode:
	case EMobilePhoneNotifyUSimAppsSelectionModeChange:
		if(iPhoneUSimApp->FoundUSimAppTags())
			{
			return iPhoneUSimApp->ExtFunc(aReqHandle, aIpc, aPckg);
			}
		else
			{
			return iPhoneSmartCardApp->ExtFunc(aReqHandle, aIpc, aPckg);
			}		

// CallBarring
	case EMobilePhoneSetCallBarringStatus:
	case EMobilePhoneSetCallBarringPassword:
	case EMobilePhoneNotifyCallBarringStatusChange:
	case EMobilePhoneGetBarringStatusPhase1:
	case EMobilePhoneGetBarringStatusPhase2:
		return iCallBarring->ExtFunc(aReqHandle, aIpc, aPckg);

// CallForwarding
	case EMobilePhoneSetCallForwardingStatus:
	case EMobilePhoneNotifyCallForwardingStatusChange:
	case EMobilePhoneGetIdentityServiceStatus:
	case EMobilePhoneGetCallForwardingStatusPhase1:
	case EMobilePhoneGetCallForwardingStatusPhase2:
		return iCallForwarding->ExtFunc(aReqHandle, aIpc, aPckg);

// CallWaiting
	case EMobilePhoneSetCallWaitingStatus:
	case EMobilePhoneNotifyCallWaitingStatusChange:
	case EMobilePhoneGetWaitingStatusPhase1:
	case EMobilePhoneGetWaitingStatusPhase2:
		{
		functionValue = 0;
		TRAPD(leaveValue, functionValue = iCallWaiting->ExtFuncL(aReqHandle, aIpc, aPckg));
		if (leaveValue != KErrNone)
	 		{
 			return leaveValue;
 			}
		else
 			{
 			return functionValue;
 			}		
		}

// IMS Auth
	case EMobilePhoneAuthorizationInfoPhase1:
			return iPhoneIMSAuth->GetAuthorizationInfoPhase1(aReqHandle,
			reinterpret_cast<RMobilePhone::TClientId*>(dataPtr),
			reinterpret_cast<TInt*>(dataPtr2));

	case EMobilePhoneAuthorizationInfoPhase2:
		return iPhoneIMSAuth->GetAuthorizationInfoPhase2(aReqHandle,
			reinterpret_cast<RMobilePhone::TClientId*>(dataPtr), aPckg.Des2n());
			
	case EMobilePhoneIMSAuthenticate:
		return iPhoneIMSAuth->GetAuthenticationData(aReqHandle, aPckg.Des1n());

	case EMobilePhoneNotifyImsAuthorizationInfoChanged:
		return iPhoneIMSAuth->NotifyImsAuthorizationInfoChanged(aReqHandle);

// Generic Smart Card Authentication
	case EMobilePhoneSmartCardAuthenticate:
		return iPhoneScAuth->GetScAuthenticationData(aReqHandle, aPckg.Des1n(),
		         reinterpret_cast<RMobilePhone::TAID*>(dataPtr2));

// Access Point Name (APN)
	case EMobilePhoneEnumerateAPNEntries:
		{
		TUint32* count = static_cast<TUint32*>(dataPtr);
		return EnumerateAPNEntries(aReqHandle,count);
		}
	
	case EMobilePhoneGetAPNname:
		{
		TUint32 index = *static_cast<TUint32*>(dataPtr);
		
		RMobilePhone::TAPNEntryV3Pckg* entryToReturnPckg = 
			static_cast<RMobilePhone::TAPNEntryV3Pckg*> (aPckg.Des2n());
		
		return GetAPNname(aReqHandle, index, *entryToReturnPckg);
		}

	case EMobilePhoneAppendAPNName:
		{
		RMobilePhone::TAPNEntryV3 entryToAppend = 
			*static_cast<RMobilePhone::TAPNEntryV3*>(dataPtr);
				
		RMobilePhone::TAPNEntryV3Pckg entryToAppendPckg(entryToAppend);	
		return AppendAPNName(aReqHandle,entryToAppendPckg);
		}
		
	case EMobilePhoneDeleteAPNName:
		{
		TUint32 tmp = *static_cast<TUint32*>(dataPtr);
		
		return DeleteAPNName(aReqHandle,tmp);
		}
	
	case EMobilePhoneNotifyAPNListChanged:
		return NotifyAPNListChanged(aReqHandle);
	
	case EMobilePhoneSetAPNControlListServiceStatus:
		return SetAPNControlListServiceStatus(aReqHandle,
				*static_cast<RMobilePhone::TAPNControlListServiceStatus*>(dataPtr));
	
	case EMobilePhoneGetAPNControlListServiceStatus:
		return GetAPNControlListServiceStatus(aReqHandle,
				*static_cast<RMobilePhone::TAPNControlListServiceStatus*>(dataPtr));
	
	case EMobilePhoneNotifyAPNControlListServiceStatusChange:
		return NotifyAPNControlListServiceStatusChange(aReqHandle,
				*static_cast<RMobilePhone::TAPNControlListServiceStatus*>(dataPtr));

	case EMobilePhoneGetServiceTable:
		return GetServiceTable(aReqHandle, 
			*static_cast<RMobilePhone::TMobilePhoneServiceTable*>(dataPtr), 
				(aPckg.Des2n()));	
	default:
		ReqCompleted(aReqHandle, KErrNotSupported);
		return KErrNone;		// Real error value returned in completion of clients request
		}
		
	}

TInt CSimPhone::CancelService(const TInt aIpc,const TTsyReqHandle aTsyReqHandle)
/**
 * Cancel an outstanding request.
 * @param aIpc The IPC number of the request that must be cancelled.  Note: this is not the
 *             IPC number of the cancel request itself.
 * @param aTsyReqHandle The TSY Request Handle of the request to be cancelled.
 */
	{
	switch(aIpc)
		{
	case EMobilePhoneNotifyIndicatorChange:
		iIndicator->NotifyIndicatorChangeCancel();
		break;

	case EMobilePhoneNotifyBatteryInfoChange:
		iBatteryCharger->NotifyBatteryInfoCancel();
		break;

	case EMobilePhoneNotifySignalStrengthChange:
		iSignalStrength->NotifySignalStrengthChangeCancel();
		break;
				
	case EMobilePhoneNotifyNetworkRegistrationStatusChange:
		iNetworkStatus->NotifyNetworkRegistrationStatusChangeCancel();
		break;

	case EMobilePhoneNotifyCurrentNetworkChange:
		iNetworkStatus->NotifyCurrentNetworkChangeCancel(aTsyReqHandle);
		break;

	case EMobilePhoneNotifyCurrentNetworkNoLocationChange:
		iNetworkStatus->NotifyCurrentNetworkNoLocationChangeCancel(aTsyReqHandle);
		break;	

	case EMobilePhoneNotifyNITZInfoChange:
		iNetworkStatus->NotifyNITZInfoChangeCancel();
		break;

	case EMobilePhoneNotifyCellInfoChange:
		iNetworkStatus->NotifyCellInfoChangeCancel(aTsyReqHandle);
		break;

	case EMobilePhoneNotifyModeChange:
		NotifyModeChangeCancel(aTsyReqHandle);
		break;


// Get Subscriber Info
	case EMobilePhoneSendDTMFTones:
		iDtmf->SendDTMFTonesCancel();
		break;

	case EMobilePhoneNotifyStopInDTMFString:
		iDtmf->NotifyStopInDTMFStringCancel();
		break;

	case EMobilePhoneNotifyDTMFCapsChange:
		iDtmf->NotifyDtmfCapsChangeCancel(aTsyReqHandle);
		break;

	case EMobilePhoneNotifySecurityCapsChange:
	case EMobilePhoneGetLockInfo:
	case EMobilePhoneNotifyLockInfoChange:
	case EMobilePhoneSetLockSetting:
	case EMobilePhoneChangeSecurityCode:
	case EMobilePhoneNotifySecurityEvent:
	case EMobilePhoneVerifySecurityCode:
	case EMobilePhoneAbortSecurityCode:
	case EMobilePhoneGetSecurityCodeInfo:
	case EMobilePhoneNotifySecurityCodeInfoChange:
		return iPhoneSecurity->CancelService(aIpc,aTsyReqHandle);

//USim Application Support
	case EMobilePhoneEnumerateUSimApplications:
	case EMobilePhoneSetUSimApplicationStatus:
	case EMobilePhoneGetUSimApplicationsInfo:
	case EMobilePhoneNotifyUSimApplicationsInfoChange:
		if(iPhoneUSimApp->FoundUSimAppTags())
			{
			return iPhoneUSimApp->CancelService(aIpc,aTsyReqHandle);
			}
		else
			{
			return iPhoneSmartCardApp->CancelService(aIpc,aTsyReqHandle);
			}
			
//SmartCard Application Support
	case EMobilePhoneEnumerateSmartCardApplications:
	case EMobilePhoneGetSmartCardApplicationInfo:
	case EMobilePhoneSetSmartCardApplicationStatus:
	case EMobilePhoneNotifySmartCardApplicationInfoChange:
	case EMobilePhoneGetScFileInfo:
	case EMobilePhoneReadScFile:
	case EMobilePhoneUpdateScFile:
		if(iPhoneSmartCardApp->FoundScAppTags())
			{
			return iPhoneSmartCardApp->CancelService(aIpc,aTsyReqHandle);
			}
		else
			{
			return CPhoneBase::CancelService(aIpc,aTsyReqHandle);;
			}

		
//USim/SmartCard Application Supoprt
	case EMobilePhoneGetCurrentActiveUSimApplication:
	case EMobilePhoneGetUSimAppsSelectionMode:
	case EMobilePhoneSetUSimAppsSelectionMode:
	case EMobilePhoneNotifyUSimAppsSelectionModeChange:
		if(iPhoneUSimApp->FoundUSimAppTags())
			{
			return iPhoneUSimApp->CancelService(aIpc,aTsyReqHandle);
			}
		else
			{
			return iPhoneSmartCardApp->CancelService(aIpc,aTsyReqHandle);
			}


		// CallBarring
	case EMobilePhoneSetCallBarringStatus:
	case EMobilePhoneSetCallBarringPassword:
	case EMobilePhoneNotifyCallBarringStatusChange:
	case EMobilePhoneGetBarringStatusPhase1:
	case EMobilePhoneGetBarringStatusPhase2:
		return iCallBarring->CancelService(aIpc, aTsyReqHandle);

		// CallForwarding
	case EMobilePhoneSetCallForwardingStatus:
	case EMobilePhoneNotifyCallForwardingStatusChange:
	case EMobilePhoneGetIdentityServiceStatus:
	case EMobilePhoneGetCallForwardingStatusPhase1:
	case EMobilePhoneGetCallForwardingStatusPhase2:
		return iCallForwarding->CancelService(aIpc, aTsyReqHandle);

		// CallWaiting
	case EMobilePhoneSetCallWaitingStatus:
	case EMobilePhoneNotifyCallWaitingStatusChange:
	case EMobilePhoneGetWaitingStatusPhase1:
	case EMobilePhoneGetWaitingStatusPhase2:
		return iCallWaiting->CancelService(aIpc, aTsyReqHandle);

		//IMS Auth
	case EMobilePhoneNotifyImsAuthorizationInfoChanged:
		return iPhoneIMSAuth->NotifyImsAuthorizationInfoChangedCancel(aTsyReqHandle);

	case EMobilePhoneAuthorizationInfoPhase1:
	case EMobilePhoneAuthorizationInfoPhase2:
		return iPhoneIMSAuth->GetAuthorizationInfoCancel(aTsyReqHandle);

	case EMobilePhoneIMSAuthenticate:
		return iPhoneIMSAuth->GetAuthenticationDataCancel(aTsyReqHandle);

		// Generic Smart Card Authenticate cancel
	case EMobilePhoneSmartCardAuthenticate:
		return iPhoneScAuth->GetScAuthenticationDataCancel(aTsyReqHandle);

// Access Point Name (APN)
	case EMobilePhoneEnumerateAPNEntries:		
	case EMobilePhoneGetAPNname:
	case EMobilePhoneAppendAPNName:
	case EMobilePhoneDeleteAPNName:
		ReqCompleted(aTsyReqHandle, KErrNone);
		return KErrNone;		
	case EMobilePhoneNotifyAPNListChanged:
		if (iAPNListNotify.iNotifyPending == TRUE)
			{
			iAPNListNotify.iNotifyPending = EFalse;
			ReqCompleted(iAPNListNotify.iNotifyHandle, KErrCancel);
			}
		return KErrNone;
	case EMobilePhoneSetAPNControlListServiceStatus:
	case EMobilePhoneGetAPNControlListServiceStatus:
		ReqCompleted(aTsyReqHandle, KErrNone);
		return KErrNone;
	case EMobilePhoneNotifyAPNControlListServiceStatusChange:		
		if (iAPNStatusNotify.iNotifyPending == TRUE)
			{
			iAPNStatusNotify.iNotifyPending = EFalse;
			ReqCompleted(iAPNStatusNotify.iNotifyHandle, KErrCancel);
			}
		return KErrNone;
	case EMobilePhoneGetServiceTable:
		ReqCompleted(aTsyReqHandle, KErrNone);
		return KErrNone;	
				
	
	case EMobilePhoneNotifyMessageWaiting:
		return iUsimRelease6->NotifyMessageWaitingCancel(aTsyReqHandle);
		
	case EMobilePhoneNotifyIccMessageWaitingIndicatorsChange:
		return iUsimRelease6->NotifyIccMessageWaitingIndicatorsChangeCancel(aTsyReqHandle);
		
	case EMobilePhoneNotifyMailboxNumbersChange:
		return iUsimRelease6->NotifyMailboxNumbersChangeCancel(aTsyReqHandle);
	
	case EMobilePhoneNotifyWlanDataChange:
		return iUsimRelease6->NotifyWlanDataChangeCancel(aTsyReqHandle);

	case EMobilePhoneNotifyPreferredWlanSIDListChange:
		return iUsimRelease6->NotifyPreferredWlanSIDListChangeCancel(aTsyReqHandle);
	
	case EMobilePhoneNotifyStorePreferredNetworksListChange:
		return iUsimRelease6->NotifyStorePreferredNetworksListChangeCancel(aTsyReqHandle);
		
	case EMobilePhoneGetPreferredNetworksPhase1:
	case EMobilePhoneGetPreferredNetworksPhase2:
		return iUsimRelease6->GetPreferredNetworksCancel(aTsyReqHandle);
		
	case EMobilePhoneGetPreferredWlanSIDsPhase1:
	case EMobilePhoneGetPreferredWlanSIDsPhase2:
		return iUsimRelease6->GetPreferredWlanSIDsCancel(aTsyReqHandle);
		
	case EMobilePhoneSetGbaBootstrapParams:
		return iUsimRelease6->SetGbaBootstrapParamsCancel(aTsyReqHandle);
		
	case EMobilePhoneNotifyAuthenticateDataChange:
		return iUsimRelease6->NotifyAuthenticateDataChangeCancel(aTsyReqHandle);
		
	case EMobilePhoneGetAuthenticationParams:
		return iUsimRelease6->GetAuthenticationParamsCancel(aTsyReqHandle);
		
	case EMobilePhoneAuthenticationListPhase1:
	case EMobilePhoneAuthenticationListPhase2:
		return iUsimRelease6->GetAuthenticationListCancel(aTsyReqHandle);

	// Default
	default:
		return CPhoneBase::CancelService(aIpc,aTsyReqHandle);
		}
	return KErrNone;
	}

CTelObject* CSimPhone::OpenNewObjectByNameL(const TDesC& aName)
/**
* This function will open a new LINE by name.
* In reality the lines should be pre-constructed in the phone's two phase constructor,
* so this will just return a pointer to the relevant line.
*
* @param aName the name of the line object to be opened
* @return CTelObject pointer to the phone object created
* @leave Leaves if incorrect phone name
*/
	{
	LOGPHONE1(">>CSimPhone::OpenNewObjectByNameL");

// Is it a voice line?
	if (aName.CompareF(KVoiceLineName) == 0)
		{
		__ASSERT_ALWAYS(iVoiceLine!=NULL,SimPanic(ELineNotPreInstantiated));
		// Base class open.
		(void)iVoiceLine->Open();
		return iVoiceLine;
		}

// Is it a data line?
	else if (aName.CompareF(KDataLineName) == 0)
		{
		__ASSERT_ALWAYS(iDataLine!=NULL,SimPanic(ESubsessionNotPreInstantiated));
		// Base class open.
		(void)iDataLine->Open();
		return iDataLine;
		}

// Is it an SMS messaging class?
	else if (aName.CompareF(KETelSmsMessaging) == 0)
		{
		__ASSERT_ALWAYS(iSmsMessaging!=NULL,SimPanic(ESubsessionNotPreInstantiated));
		// Base class open.
		(void)iSmsMessaging->Open();
		return iSmsMessaging;
		}

	else if (aName.CompareF(KPacketName) == 0)
		{
		__ASSERT_ALWAYS(iPacketService!=NULL,SimPanic(ESubsessionNotPreInstantiated));
		// Base class open.
		(void)iPacketService->Open();
		return iPacketService;
		}
	else if (aName.CompareF(KSatName) == 0)
		{
		__ASSERT_ALWAYS(iSat!=NULL,SimPanic(ESubsessionNotPreInstantiated));
		// Base class open.
		(void)iSat->Open();
		return iSat;
		}
	else if (aName.Left(SCEAP_SSN_LENGTH).Compare(KETelSmartCardEapSession) == 0)
		{
		// "num" is used for both lengths and error codes!  saves on stack! :)

		// Extracting AID
		RMobilePhone::TAID tempAID;
		// get number of AID bytes (each byte uses two chars)
		TInt num = CharToSeptNumL(aName[SCEAP_SSN_LENGTH]) * 2;
		// get the segment containing the AID
		TPtrC ptrAid = aName.Mid(SCEAP_SSN_LENGTH + 1, num);
		// convert AID to its correct binary value
		ConvertTextToBinary(ptrAid, tempAID);

		// Extracting EapType
		RMobileSmartCardEap::TEapType eapType;
		// get position of EAP type's length
		TInt posTypeLength = SCEAP_SSN_LENGTH + num + 1;
		// get the value of the length of EAP type
		num = CharToSeptNumL(aName[posTypeLength]);
		// get the segment containing the EAP type
		TPtrC ptrEapT = aName.Mid(posTypeLength + 1, num);
		eapType.Copy(ptrEapT);

		CTelObject* phoneScEap = iPhoneScEapMan->CreateScEapSubSessionL(tempAID, eapType);

		if (phoneScEap == NULL)
			{
			LOGPHONE1("ERROR CSimSmartCardEap object not created, returning KErrGeneral");
			User::Leave(KErrGeneral);
			}

		LOGPHONE2("CSimPhone::OpenNewObjectByNameL CSimSmartCardEap object created [0x%08x]", phoneScEap);
		return phoneScEap;
		} // End of opening Smart Card EAP sub-session
	else if (aName.CompareF(KETelOwnNumberStore) == 0)
		{
		if(iONStore==NULL)
			{
			LOGPHONE1("ERROR CSimONStore object not found. Please check config file.");
			User::Leave(KErrNotFound);
			}
		else
			{
			(void)iONStore->Open();
			return iONStore;
			}
		}
// Is it a phonebook store?
	else
		{
		TInt i = 0;
		TBuf8<KMaxName> name;
		TInt tLength;
		TPtrC remain(aName);
		name.Copy(aName);		// Do simple 16 bit to 8 bit conversion

		if ((tLength=remain.Find(PHBOOK_DELIMITER))==KErrNotFound)
			{
			for(i=0;i<iPhBkStores->Count();i++)
				{
				if(name.MatchF(iPhBkStores->At(i)->Name())==0)
					{
					// Base class open.
					(void)iPhBkStores->At(i)->Open();
					return iPhBkStores->At(i);
					}
				}
			for(i=0;i<iPhBkUSimStores->Count();i++)
				{
				if((name.MatchF(iPhBkUSimStores->At(i)->Name())==0) &&
					(iPhBkUSimStores->At(i)->PhBkStore().Compare(KGsmPhoneBook)==0))

					{
					(void)iPhBkUSimStores->At(i)->Open();
					return iPhBkUSimStores->At(i);
					}

				}
			}
		else
			{
			TBuf8<KMaxName> PhBkName;
			TBuf8<KMaxName> PhBkStore;
			PhBkName.Copy(aName.Left(tLength));
			PhBkStore.Copy(aName.Right(aName.Length()-tLength-2));

			for(TInt i=0;i<iPhBkUSimStores->Count();i++)
				{
				if((PhBkName.MatchF(iPhBkUSimStores->At(i)->Name())==0) &&
					(PhBkStore.MatchF(iPhBkUSimStores->At(i)->PhBkStore())==0))

					{
					(void)iPhBkUSimStores->At(i)->Open();
					return iPhBkUSimStores->At(i);
					}

				}
			}
		}
// Not matched...
	User::Leave(KErrNotFound);
	return NULL;
	}


CTelObject* CSimPhone::OpenNewObjectL(TDes& /*aNewName*/)
/**
* This function would open a new unnamed object,
* but this functionality is not supported at the phone level.
* All lines must be opened by name.
*
* @param aNewName a reference to the name of the phone object created
* @return CTelObject a pointer to the phone object created
* @leave Leaves if no memory is available
*/
	{
	User::Leave(KErrNotSupported);
	return NULL;
	}

CTelObject::TReqMode CSimPhone::ReqModeL(const TInt aIpc)
/**
* This function returns the Request Mode for the request with the passed IPC value.
* The ETel Server provides a function for returning the standard request modes for
* the Core API requests.
*/
	{
	TReqMode reqMode=0;
	switch(aIpc)
		{
	case EMobilePhoneGetPhoneStoreInfo:
	case EMobilePhoneGetSignalCaps:
	case EMobilePhoneGetSignalStrength:
	case EMobilePhoneGetNetworkCaps:
	case EMobilePhoneGetCurrentMode:

	case EMobilePhoneNotifyModeChange:

	case EMobilePhoneGetHomeNetwork:
	case EMobilePhoneGetCurrentNetwork:
	case EMobilePhoneGetCurrentNetworkNoLocation:
	case EMobilePhoneGetNetworkRegistrationStatus:
	case EMobilePhoneGetNITZInfo:
	case EMobilePhoneGetCellInfo:
	case EMobilePhoneGetSubscriberId:
	case EMobilePhoneGetDTMFCaps:
	case EMobilePhoneSendDTMFTones:
	case EMobilePhoneStartDTMFTone:
	case EMobilePhoneStopDTMFTone:
	case EMobilePhoneContinueDTMFStringSending:
	case EMobilePhoneGetMultimodeCaps:
	case EMobilePhoneGetIndicator:
	case EMobilePhoneGetIndicatorCaps:
	case EMobilePhoneGetBatteryInfo:
	case EMobilePhoneGetBatteryCaps:
	case EMobilePhoneGetIccAccessCaps:
	case EMobilePhoneGetSecurityCaps:
	case EMobilePhoneGetLockInfo:
	case EMobilePhoneSetLockSetting:
	case EMobilePhoneGetSecurityCodeInfo:
	case EMobilePhoneChangeSecurityCode:
	case EMobilePhoneVerifySecurityCode:
	case EMobilePhoneAbortSecurityCode:
	case EMobilePhoneEnumerateUSimApplications:
	case EMobilePhoneGetCurrentActiveUSimApplication:
	case EMobilePhoneSetUSimApplicationStatus:
	case EMobilePhoneGetUSimApplicationsInfo:
	case EMobilePhoneGetUSimAppsSelectionMode:
	case EMobilePhoneSetUSimAppsSelectionMode:
	case EMobilePhoneEnumerateSmartCardApplications:
	case EMobilePhoneGetSmartCardApplicationInfo:
	case EMobilePhoneSetSmartCardApplicationStatus:
	case EMobilePhoneGetScFileInfo:
	case EMobilePhoneReadScFile:
	case EMobilePhoneUpdateScFile:
	case EMobilePhoneSetCallBarringStatus:
	case EMobilePhoneSetCallBarringPassword:
	case EMobilePhoneGetBarringStatusPhase1:
	case EMobilePhoneGetBarringStatusPhase2:
	case EMobilePhoneSetCallForwardingStatus:
	case EMobilePhoneGetIdentityServiceStatus:
	case EMobilePhoneGetCallForwardingStatusPhase1:
	case EMobilePhoneGetCallForwardingStatusPhase2:
	case EMobilePhoneGetIdentityCaps:
	case EMobilePhoneGetPhoneId:
	case EMobilePhoneGetServiceProviderName:
	case EMobilePhoneGetNetworkName:
	case EMobilePhoneSetCallWaitingStatus:
	case EMobilePhoneGetWaitingStatusPhase1:
	case EMobilePhoneGetWaitingStatusPhase2:
	case EMobilePhoneAuthorizationInfoPhase1:
	case EMobilePhoneAuthorizationInfoPhase2:
	case EMobilePhoneIMSAuthenticate:
	case EMobilePhoneSmartCardAuthenticate:
	case EMobilePhoneGetMailboxNumbers:
	case EMobilePhoneGetIccMessageWaitingIndicators:
	case EMobilePhoneSetIccMessageWaitingIndicators:
	case EMobilePhoneGetWlanData:
	case EMobilePhoneSetWlanData:
	case EMobilePhoneGetPreferredWlanSIDsPhase1:
	case EMobilePhoneGetPreferredWlanSIDsPhase2:
	case EMobilePhoneStorePreferredWlanSIDList:	
	case EMobilePhoneGetPreferredNetworksPhase1:
	case EMobilePhoneGetPreferredNetworksPhase2:
	case EMobilePhoneStorePreferredNetworksList:
	case EMobilePhoneSetGbaBootstrapParams:
	case EMobilePhoneGetAuthenticationParams:
	case EMobilePhoneNotifyAuthenticateDataChange:
	case EMobilePhoneAuthenticationListPhase1:
	case EMobilePhoneAuthenticationListPhase2:
		break;

// Service Tables
	case EMobilePhoneEnumerateAPNEntries:
	case EMobilePhoneGetAPNname:
	case EMobilePhoneAppendAPNName:
	case EMobilePhoneDeleteAPNName:
	case EMobilePhoneNotifyAPNListChanged:
	case EMobilePhoneSetAPNControlListServiceStatus:
	case EMobilePhoneGetAPNControlListServiceStatus:
	case EMobilePhoneNotifyAPNControlListServiceStatusChange:
	case EMobilePhoneGetServiceTable:
		break;

	case EMobilePhoneNotifyIndicatorChange:
	case EMobilePhoneNotifyBatteryInfoChange:
	case EMobilePhoneNotifySignalStrengthChange:
	case EMobilePhoneNotifyNetworkRegistrationStatusChange:
	case EMobilePhoneNotifyCurrentNetworkChange:
	case EMobilePhoneNotifyCurrentNetworkNoLocationChange:
	case EMobilePhoneNotifyNITZInfoChange:
	case EMobilePhoneNotifyCellInfoChange:
	case EMobilePhoneNotifyDTMFCapsChange:
	case EMobilePhoneNotifyStopInDTMFString:
	case EMobilePhoneNotifySecurityEvent:
	case EMobilePhoneNotifySecurityCapsChange:
	case EMobilePhoneNotifyUSimApplicationsInfoChange:
	case EMobilePhoneNotifySmartCardApplicationInfoChange:
	case EMobilePhoneNotifyUSimAppsSelectionModeChange:
	case EMobilePhoneNotifyCallBarringStatusChange:
	case EMobilePhoneNotifyCallWaitingStatusChange:
	case EMobilePhoneNotifyCallForwardingStatusChange:
	case EMobilePhoneNotifyImsAuthorizationInfoChanged:
	case EMobilePhoneNotifyMailboxNumbersChange:
	case EMobilePhoneNotifyIccMessageWaitingIndicatorsChange:
	case EMobilePhoneNotifyMessageWaiting:
	case EMobilePhoneNotifyWlanDataChange:
	case EMobilePhoneNotifyPreferredWlanSIDListChange:
	case EMobilePhoneNotifyStorePreferredNetworksListChange:
		reqMode=KReqModeMultipleCompletionEnabled | KReqModeRePostImmediately;
		break;
	case EMobilePhoneNotifyLockInfoChange:
	case EMobilePhoneNotifySecurityCodeInfoChange:
		reqMode=KReqModeMultipleCompletionEnabled;
		break;
	default:
		reqMode=CPhoneBase::ReqModeL(aIpc);
		break;
		}
	return reqMode;
	}

TInt CSimPhone::RegisterNotification(const TInt /*aIpc*/)
/**
* The ETel Server calls this function when the first client makes a notification request.
* If supported by the underlying protocol controlling the signalling stack,
* this can be used to start requesting updates for the relevant	service.
* This function does nothing in this TSY.
*
* @param aIpc The Ipc representing the client notification request
* @return KErrNone
*/
	{
	return KErrNone;
	}

TInt CSimPhone::DeregisterNotification(const TInt /*aIpc*/)
/**
* The ETel Server calls this function when the last client that had previously
* made a notification request closes its ETel Server handle.  If supported by
* the underlying protocol controlling the	signalling stack, this can be used
* to stop requesting updates for the relevant service.
*
* @param aIpc The Ipc representing the client notification request
* @return KErrNone
*/
	{
	return KErrNone;
	}

TInt CSimPhone::NumberOfSlotsL(const TInt aIpc)
	{
	switch(aIpc)
		{
	case EMobilePhoneNotifyIndicatorChange:
	case EMobilePhoneNotifyBatteryInfoChange:
	case EMobilePhoneNotifySignalStrengthChange:
	case EMobilePhoneNotifyCurrentNetworkChange:
	case EMobilePhoneNotifyCurrentNetworkNoLocationChange:
	case EMobilePhoneNotifyNetworkRegistrationStatusChange:
	case EMobilePhoneNotifyNITZInfoChange:
	case EMobilePhoneNotifyCellInfoChange:
	case EMobilePhoneNotifyStopInDTMFString:
	case EMobilePhoneNotifySecurityEvent:
	case EMobilePhoneNotifyUSimApplicationsInfoChange:
	case EMobilePhoneNotifySmartCardApplicationInfoChange:
	case EMobilePhoneNotifyUSimAppsSelectionModeChange:
	case EMobilePhoneNotifySecurityCapsChange:
	case EMobilePhoneNotifyCallBarringStatusChange:
	case EMobilePhoneNotifyCallForwardingStatusChange:
	case EMobilePhoneNotifyCallWaitingStatusChange:
	case EMobilePhoneNotifyImsAuthorizationInfoChanged:
	case EMobilePhoneNotifyMailboxNumbersChange:
	case EMobilePhoneNotifyIccMessageWaitingIndicatorsChange:
	case EMobilePhoneNotifyMessageWaiting:
	case EMobilePhoneNotifyWlanDataChange:
	case EMobilePhoneNotifyPreferredWlanSIDListChange:
	case EMobilePhoneNotifyStorePreferredNetworksListChange:
		return KDefaultNumberOfSlots;
	default:
		break;
		}
	return CPhoneBase::NumberOfSlotsL(aIpc);
	}

void CSimPhone::Init()
/**
*	This function can be used to perform any necessary synchronous initialisation.
*/
	{
	}

TInt CSimPhone::ControlledInitialisation(const TTsyReqHandle aTsyReqHandle)
/**
* This function can be used to start any necessary asynchronous initialisation.
* In this TSY, there are no asynchronous initialisation procedures, so its empty.
*/
	{
	iSimPhoneInitialise->Start(aTsyReqHandle);
	return KErrNone;
	}

TInt CSimPhone::ControlledInitialisationCancel(const TTsyReqHandle /*aTsyReqHandle*/)
/**
* Request to cancel the ::ControlledInitialisation function.
* Since this TSY has no asynchronous initialisation procedures,
* this function should never be called.
*/
	{
	iSimPhoneInitialise->Cancel();
	return KErrNone;
	}

TInt CSimPhone::NotifyCapsChange(const TTsyReqHandle aTsyReqHandle, RPhone::TCaps* /*aCaps*/)
/**
* The phone caps will not change dynamically in this TSY as it is hardcoded,
* so there is no chance of completing this request.
*
* @param aTsyReqHandle
* @param aCaps a pointer to the phone capability
* @return KErrNone
*/
	{
	ReqCompleted(aTsyReqHandle, KErrNotSupported);
	return KErrNone;
	}

TInt CSimPhone::NotifyCapsChangeCancel(const TTsyReqHandle /*aTsyReqHandle*/)
/**
*	Cancel the NotifyCapsChange request.
*
* @param aTsyReqHandle
* @return KErrNone
*/
	{
	return KErrNone;
	}



TInt CSimPhone::NotifyModemDetected(const TTsyReqHandle aTsyReqHandle, RPhone::TModemDetection* /*aDetection*/)
/**
* This request will be completed when the phone's connection status changes.
* This function is not supported in this version of the TSY
*/
	{
	ReqCompleted(aTsyReqHandle,KErrNotSupported);
	return KErrNone;
	}

TInt CSimPhone::NotifyModemDetectedCancel(const TTsyReqHandle /*aTsyReqHandle*/)
/**
* Cancel outstanding modem detection notification, by TSY handle
* Returns KErrNone
*/
	{
	return KErrNone;
	}

TInt CSimPhone::GetInfo(const TTsyReqHandle aTsyReqHandle, RPhone::TPhoneInfo* aPhoneInfo)
/**
* Retrieve the Phone Information
* It should always return TModemDetection::EDetectedPresent
*
* @param aTsyReqHandle
* @param aPhoneInfo pointer to the phone information to be returned to client
* @return KErrNone
*/
	{
	aPhoneInfo->iDetection=iModemDetection;
	ReqCompleted(aTsyReqHandle,KErrNone);
	return KErrNone;
	}

void CSimPhone::UpdateNetworkMode(RMobilePhone::TMobilePhoneNetworkMode aNetworkMode)
	{
	iNtwkMode = aNetworkMode;
	}

RMobilePhone::TMobilePhoneNetworkMode CSimPhone::NetworkMode()
	{
	return iNtwkMode;
	}

TInt CSimPhone::GetCurrentMode(TTsyReqHandle aReqHandle,TDes8* aPckg1)
/**
 * Retrieve the Current Mode.
 * @param aReqHandle	The request handle associated with this request.
 * @param aPckg1		The first parameter package.  This will be populated with the
 *						mode setting to be returned.
 * @return TInt			Standard error value.
 */
	{
	TPckg<RMobilePhone::TMobilePhoneNetworkMode>* modePckg=(TPckg<RMobilePhone::TMobilePhoneNetworkMode>*)aPckg1;
	RMobilePhone::TMobilePhoneNetworkMode& mode=(*modePckg)();

	mode=iNtwkMode;
	ReqCompleted(aReqHandle,KErrNone);
	return KErrNone;
	}

TInt CSimPhone::GetCaps(const TTsyReqHandle aTsyReqHandle, RPhone::TCaps* aCaps)
/**
* Retrieve the phone capabilities
*
* @param aTsyReqHandle
* @param aCaps a pointer to the Phone capability
* @return KErrNone
*/
	{
	aCaps->iFlags= RPhone::KCapsVoice | RPhone::KCapsData;
	ReqCompleted(aTsyReqHandle,KErrNone);
	return KErrNone;
	}

TInt CSimPhone::GetStatus(const TTsyReqHandle aTsyReqHandle, RPhone::TStatus* aStatus)
/**
* Retrieve the phone status.
*
* @param aTsyReqHandle
* @param aStatus pointer to the phone status info.
* @return KErrNone
*/
	{
	aStatus->iModemDetected=iModemDetection;
	aStatus->iMode=iMode;
	ReqCompleted(aTsyReqHandle,KErrNone);
	return KErrNone;
	}

TInt CSimPhone::EnumerateLines(const TTsyReqHandle aTsyReqHandle, TInt* aParams)
/**
* Enumerate the lines
* Returns 2 (for the voice and data lines) in the integer reference passed in
* through its function argument.
*
* @param aTsyReqHandle
* @param aParams pointer to the number of lines supported
* @return KErrNone
*/
	{
	*aParams = KNumberOfLines;
	ReqCompleted(aTsyReqHandle,KErrNone);
	return KErrNone;
	}

TInt CSimPhone::GetLineInfo(const TTsyReqHandle aTsyReqHandle, TLineInfoIndex* aParams)
/**
* Gets the line info on the particular CSimLine object required.
* Returns the current hook status, the current line status, name of last call created on
* that line and the name of the call to which a new incoming call will be directed.
* TLineInfoIndex specifies which lines' info is requested.
*
* @param aTsyReqHandle
* @param aParam pointer to the line info
* @return KErrNone
*/
	{
	if (aParams->iIndex==KVoiceLineIndex)
		{
		aParams->iInfo.iStatus = (RCall::TStatus)iVoiceLine->iState;
		aParams->iInfo.iName = iVoiceLine->iLineName;
		aParams->iInfo.iLineCapsFlags = (RLine::KCapsVoice|RLine::KCapsEventIncomingCall);
		ReqCompleted(aTsyReqHandle,KErrNone);
		}
	else if (aParams->iIndex==KDataLineIndex)
		{
		aParams->iInfo.iStatus = (RCall::TStatus)iDataLine->iState;
		aParams->iInfo.iName = iDataLine->iLineName;
		aParams->iInfo.iLineCapsFlags = (RLine::KCapsData|RLine::KCapsEventIncomingCall);
		ReqCompleted(aTsyReqHandle,KErrNone);
		}
	else
		{
		ReqCompleted(aTsyReqHandle,KErrNotFound);
		}
	return KErrNone;
	}

TInt CSimPhone::GetPhoneStoreInfo(TTsyReqHandle aReqHandle,TDes8* aPckg1,TDes16* aPckg2)
/**
 * Retrieve information about a named Phonebook Store.
 * @param aReqHandle	The TSY request handle associated with this request.
 * @param aPckg1		The parameter package containing the index of the SMS Store for which
 *						information is going to be retrieved.
 * @param aPckg2		The parameter package in which the retrieved SMS Store information will
 *						be passed back to the client.
 * @return TInt			Standard error value.
 */
	{
	TPckg<RMobilePhoneStore::TMobilePhoneStoreInfoV1>* infoPckg=(TPckg<RMobilePhoneStore::TMobilePhoneStoreInfoV1>*)aPckg1;
	RMobilePhoneStore::TMobilePhoneStoreInfoV1& info=(*infoPckg)();

	// Check that the data structure is supported by the simulated TSY version
	TInt err = CheckSimTsyVersion(info);
	if(err != KErrNone)
		{
		ReqCompleted(aReqHandle, err);
		return KErrNone;
		}

	TInt i;
	for(i=0;i<iPhBkStores->Count();i++)
		{
		TBuf8<KMaxName> nBuf;
		nBuf.Copy(*aPckg2);
		if(iPhBkStores->At(i)->Name().MatchF(nBuf)==0)
			break;
		}

	if(i==iPhBkStores->Count())
		{
		ReqCompleted(aReqHandle,KErrNotFound);
		return KErrNone;
		}

	iPhBkStores->At(i)->PopulatePhBkStoreInfo(&info);
	ReqCompleted(aReqHandle,KErrNone);
	return KErrNone;
	}

TInt CSimPhone::GetPhoneStoreInfo(TTsyReqHandle aReqHandle,TDes8* aPckg1,
								  RMobilePhone::TPhoneStoreNameAndIccType* aTable)
/**
 * Retrieve information about a named Phonebook Store.
 * @param aReqHandle	The TSY request handle associated with this request.
 * @param aPckg1		The parameter package containing the index of the PhoneBook Store for which
 *						information is going to be retrieved.
 * @param aTable		The parameter package in which the retrieved PhoneBook Store name and type information will
 *						be passed from the client.
 * @return TInt			Standard error value.
 */
	{
	TBool found = EFalse;
	TPckg<RMobilePhoneStore::TMobilePhoneStoreInfoV1>* infoPckg=(TPckg<RMobilePhoneStore::TMobilePhoneStoreInfoV1>*)aPckg1;
	RMobilePhoneStore::TMobilePhoneStoreInfoV1& info=(*infoPckg)();

	// Check that the data structure is supported by the simulated TSY version
	TInt err = CheckSimTsyVersion(info);
	if(err != KErrNone)
		{
		ReqCompleted(aReqHandle, err);
		return KErrNone;
		}

	TBuf8<KMaxName> n1Buf,n2Buf;

	n1Buf.Copy(aTable->iStoreName);
	n2Buf.Copy(aTable->iMode);

	TInt i;

	for(i=0;i<iPhBkUSimStores->Count();i++)
		{
		if((iPhBkUSimStores->At(i)->Name().MatchF(n1Buf)==0) &&
			(iPhBkUSimStores->At(i)->PhBkStore().MatchF(n2Buf)==0))
			{
			found = ETrue;
			break;
			}
		}

	if(!found)
		{
			for(i=0;i<iPhBkStores->Count();i++)
				{
				if((iPhBkStores->At(i)->Name().MatchF(n1Buf)==0)  &&
					n2Buf != KUSimPhoneBook)
					{
					found = ETrue;
					break;
					}
				}
			if(!found)
				{
				ReqCompleted(aReqHandle,KErrNotFound);
				return KErrNone;
				}
			else
				{
				iPhBkStores->At(i)->PopulatePhBkStoreInfo(&info);
				ReqCompleted(aReqHandle,KErrNone);
				}
		}
	else
		{
		if(iPhBkUSimStores->At(i)->CheckAndSwitchUSimApps()!=KErrNone)
			{
			ReqCompleted(aReqHandle, KErrNotFound);
			return KErrNone;
			}

		iPhBkUSimStores->At(i)->PopulatePhBkStoreInfo(&info);
		ReqCompleted(aReqHandle,KErrNone);
		}
	return KErrNone;
	}

TInt CSimPhone::GetMultimodeCaps(const TTsyReqHandle aTsyReqHandle,TUint32* aCaps)
/**
 * Retrieve the multimode capabilities of this TSY.  This is hardcoded to support GSM, GPRS and CDMA.
 * @param aTsyReqHandle		TSY Request Handle associated with this request.
 * @param aCaps				The multimode capability structure to be populated with the result.
 * @return TInt				Standard error value.
 */
	{
	*aCaps = RMobilePhone::KCapsGsmSupported |
	         RMobilePhone::KCapsGprsSupported |
	         RMobilePhone::KCapsEapSupported | 
	         RMobilePhone::KCapsWcdmaSupported;

	ReqCompleted(aTsyReqHandle,KErrNone);

	return KErrNone;
	}

TInt CSimPhone::GetIccAccessCaps(const TTsyReqHandle aTsyReqHandle,TUint32* aCaps)
/**
 * Retrieve the access capabilities of this ICC using SimTSY.  This is hardcoded to support SIM, RUIM and USIM.
 * @param aTsyReqHandle		TSY Request Handle associated with this request.
 * @param aCaps				The multimode capability structure to be populated with the result.
 * @return TInt				Standard error value.
 */
	{
	*aCaps = RMobilePhone::KCapsSimAccessSupported | RMobilePhone::KCapsRUimAccessSupported | RMobilePhone::KCapsUSimAccessSupported;
	ReqCompleted(aTsyReqHandle,KErrNone);

	return KErrNone;
	}

TInt CSimPhone::GetSubscriberInfo(TTsyReqHandle aReqHandle,TDes8* aPckg1)
/**
 * Retrieve the Subscriber Identifier.
 * @param aReqHandle	The TSY request handle associated with this request.
 * @param aPckg1		The first parameter package.  This will be populated with the
 *						subscriber information.
 * @return TInt			Standard error value.
 */
	{
	TPckg<RMobilePhone::TMobilePhoneSubscriberId>* subscribePckg=(TPckg<RMobilePhone::TMobilePhoneSubscriberId>*)aPckg1;
	RMobilePhone::TMobilePhoneSubscriberId& subscribe=(*subscribePckg)();
	TInt ret = KErrNone;
	if (iSubscriberId.iError == KErrNone)
		{
		if(iSubscriberId.iIMSI.Length()>RMobilePhone::KIMSISize)
		    {
			subscribe.Copy(iSubscriberId.iIMSI.Left(RMobilePhone::KIMSISize));
		    }
		else
		    {
			subscribe.Copy(iSubscriberId.iIMSI);
		    }
		}
	else
	    {
		ret = iSubscriberId.iError;
	    }
	ReqCompleted(aReqHandle,ret);
	return KErrNone;
	}

const CTestConfigSection* CSimPhone::CfgFile()
/**
*	Return a pointer to the Configuration File Section
*
* @return CTestConfigSection	pointer to the configuration file section
*/
	{
	return iConfigSection;
	}

const CTestConfigSection* CSimPhone::DefaultCfgFile()
/**
*	Return a pointer to the Default Configuration File Section
*
* @return CTestConfigSection	pointer to the default configuration file section
*/
	{
	if( iConfigFile == NULL )
	    {
	    return NULL;
	    }
	return iConfigFile->Section(KScriptDefaults);
	}

TInt CSimPhone::ValidateChangeState(CSimLine* aOriginatingLine, RMobileCall::TMobileCallStatus aState)
/**
*	This function validates the proposed state change by using the iActiveLine
*	pointer member variable.  If the proposed change of state would clash with
*	an existing call, the change is	errored.
*	The function also updates the iMode state to reflect the new call state.
*
* @param aOriginatingLine pointer to the line requesting for the change of state
* @param aState the state the aOriginatingLine wants to change to
* @return Error indicates whether the change of state is successful or not
*/
	{
	__ASSERT_ALWAYS(iMode!=RPhone::EModeUnknown,SimPanic(EPhoneModeUnknownIllegal));
	__ASSERT_ALWAYS(iNtwkMode!=RMobilePhone::ENetworkModeUnknown,SimPanic(ENetworkModeUnknownIllegal));

// If there is no active line defined, then any state changes are fine.  However,
// we need to watch for a shift to an "active" status.
	LOGPHONE1(">>CSimPhone::ValidateChangeState");
	if(!iActiveLine)
		{
		iMode=ConvertStateToMode(aState);
		if(IsStateActive(aState))
			iActiveLine=aOriginatingLine;
		// If a call is in progress, we may need to trigger NotifyIndicatorChange
		CheckIndicatorNotification();
		return KErrNone;
		}

// If this state change request is coming from the active line then any state changes
// are ok, but we must watch for a change that would "deactivate" the line.
	if(iActiveLine==aOriginatingLine)
		{
		iMode=ConvertStateToMode(aState);
		if(!IsStateActive(aState))
			iActiveLine=NULL;
		// If a call is in progress, we may need to trigger NotifyIndicatorChange
		CheckIndicatorNotification();
		return KErrNone;
		}

// If we have got this far, then there is an active line, but this state change
// request has not come from it.  If this is the case, then a change to ringing
// or Idle are the only valid state changes.

	if((aState==RMobileCall::EStatusRinging) ||
	   (aState==RMobileCall::EStatusIdle))
		return KErrNone;

	LOGPHONE1("<<CSimPhone::ValidateChangeState");
	return KErrGeneral;
	}

TInt CSimPhone::FindActiveVoiceCall(CSimVoiceCall*& aCall)
/**
 * Find an active voice call.  Return KErrNotFound if no voice calls active.
 * @param aCall		Pointer to active voice call.
 * @return TInt		Standard return error.
 */
	{
	return iVoiceLine->FindActiveVoiceCall(aCall);
	}

TInt CSimPhone::ValidateChangeState(RPacketService::TStatus aState)
/**
*	This function validates the proposed state change by using the iActiveLine
*	pointer member variable.  If the proposed change of state would clash with
*	an existing call, the change is	errored.
*	The function also updates the iMode state to reflect the new call state.
*
* @param aOriginatingLine pointer to the line requesting for the change of state
* @param aState the state the aOriginatingLine wants to change to
* @return Error indicates whether the change of state is successful or not
*/
	{
	LOGPHONE1(">>CSimPhone::ValidateChangeState packet");
	__ASSERT_ALWAYS(iMode!=RPhone::EModeUnknown,SimPanic(EPhoneModeUnknownIllegal));
	__ASSERT_ALWAYS(iNtwkMode!=RMobilePhone::ENetworkModeUnknown,SimPanic(ENetworkModeUnknownIllegal));
	__ASSERT_ALWAYS(iPacketService->MSClass()!=RPacketService::EMSClassUnknown,SimPanic(EPacketMSClassUnKnown));

//we must know the type of class the phone supports in order to determine what action to take.
	switch(iPacketService->MSClass())
		{
	case RPacketService::EMSClassDualMode:				//< Active Simultaneous PS & CS calls supported (ClassA)
		return KErrNone;
	case RPacketService::EMSClassSuspensionRequired:	//< Active CS & Suspended PS simultaneous calls supported (ClassB)
		if(iMode == RPhone::EModeIdle)
			return KErrNone;
		else if((aState == RPacketService::EStatusUnattached) || (aState == RPacketService::EStatusSuspended))
			return KErrNone;
		else return KErrNotSupported; //Should actually force a suspend of the packet context and service
	case RPacketService::EMSClassAlternateMode:			//< Active CS or Active PS only call supported (ClassC)
		if(iMode == RPhone::EModeIdle)
			return KErrNone;
		else if(aState == RPacketService::EStatusUnattached)
			return KErrNone;
		else return KErrNotSupported;
	case RPacketService::EMSClassCircuitSwitchedOnly:	//< Active CS only call supported (ClassC)
		return KErrNotSupported; //Should actually force a detach from the network if not in unattached state already
	case RPacketService::EMSClassPacketSwitchedOnly:	//< Active PS only call supported (ClassC)
		return KErrNone; //Should disconnect all active calls if mode is not Idle.
	default:
		return KErrNotSupported;
		}
	}

RMobilePhone::TMobilePhoneBatteryStatus CSimPhone::BatteryStatus()
	{
	return iBatteryCharger->Status();
	}

RMobilePhone::TMobilePhoneRegistrationStatus CSimPhone::RegistrationStatus()
	{
	return iNetworkStatus->RegistrationStatus();
	}

void CSimPhone::CheckIndicatorNotification()
	{
	iIndicator->CheckNotification();
	}

/**
* A utility function for checking whether the first pin on the phone 
* requires authentication 
*
* @return TBool true if required and false if not 
*/
TBool CSimPhone::IsICCLocked()
	{
	return iPhoneSecurity->IsICCLocked();
	}

/**
* A utility function for checking whether the second pin on the phone 
* requires authentication 
*
* @return TBool true if required and false if not 
*/
TBool CSimPhone::IsPIN2Locked()
	{
	return iPhoneSecurity->IsPIN2Locked();
	}

TBool CSimPhone::IsHiddenEnabled()
	{
	return iPhoneSecurity->IsHiddenEnabled();
	}

void CSimPhone::SecurityEvent(RMobilePhone::TMobilePhoneSecurityEvent aEvent)
	{
	iPhoneSecurity->SecurityEvent(aEvent);
	}

RMobilePhone::TAID CSimPhone::GetActiveUSim()
	{
	if(iPhoneUSimApp->FoundUSimAppTags())
		{
		return iPhoneUSimApp->GetActiveUSim();
		}
	else
		{
		return iPhoneSmartCardApp->GetActiveUSim();
		}
	
	}

RMobilePhone::TMobilePhoneSubscriberId CSimPhone::GetImsi()
	{
	return iSubscriberId.iIMSI;
	}
void CSimPhone::GetPhoneIdAndCaps()
	{
	TPtrC8 imei, model, manufacturer, revision;
	const CTestConfigItem* item=CfgFile()->Item(KPhoneId);
	if (item)
		{
		TInt ret=CTestConfig::GetElement(item->Value(),KStdDelimiter,0,manufacturer);
		if(ret!=KErrNone)
			LOGPARSERR("manufacturer",ret,0,&KPhoneId);
		ret = CTestConfig::GetElement(item->Value(),KStdDelimiter,1,model);
		if(ret!=KErrNone)
			LOGPARSERR("model",ret,1,&KPhoneId);
		ret = CTestConfig::GetElement(item->Value(),KStdDelimiter,2,revision);
		if(ret!=KErrNone)
			LOGPARSERR("revision",ret,2,&KPhoneId);
		ret = CTestConfig::GetElement(item->Value(),KStdDelimiter,3,imei);
		if(ret!=KErrNone)
			LOGPARSERR("imei",ret,3,&KPhoneId);
		ret = CTestConfig::GetElement(item->Value(),KStdDelimiter,4,iPhoneId.iError);

		iPhoneId.iManufacturerId.Copy(manufacturer);
		iPhoneId.iModelId.Copy(model);
		iPhoneId.iRevisionId.Copy(revision);
		iPhoneId.iSerialId.Copy(imei);
		}
	else
		{
		//set to default
		iPhoneId.iManufacturerId.Copy(KPhoneManufacturerDefault);
		iPhoneId.iModelId.Copy(KPhoneModelDefault);
		iPhoneId.iRevisionId.Copy(KPhoneRevesionDefault);
		iPhoneId.iSerialId.Copy(KPhoneSerialNumberDefault);
		iPhoneId.iError = KErrNone;
		}

	//Set Phone Identity Caps
	iPhoneIdCaps = 0;
	if(iPhoneId.iManufacturerId.Length() > 0)
		iPhoneIdCaps |= RMobilePhone::KCapsGetManufacturer;

	if(iPhoneId.iModelId.Length() > 0)
		iPhoneIdCaps |=  RMobilePhone::KCapsGetModel;

	if(iPhoneId.iRevisionId.Length() > 0)
		iPhoneIdCaps |=  RMobilePhone::KCapsGetRevision;

	if(iPhoneId.iSerialId.Length() > 0)
		iPhoneIdCaps |=  RMobilePhone:: KCapsGetSerialNumber;

	if(iSubscriberId.iIMSI.Length() > 0)
		iPhoneIdCaps |=  RMobilePhone::KCapsGetSubscriberId;
	}

TInt CSimPhone::GetIdentityCaps(TTsyReqHandle aReqHandle, TUint32& aCaps)
	{
	aCaps = iPhoneIdCaps;
	ReqCompleted(aReqHandle,KErrNone);
	return KErrNone;
	}
	
/**
* A utility function for checking whether a service is active in the service table
*
* @param aTable pointer to the table to be tested for active service
* @param aOffset to pointer in the service table
* @param aModifier which service to be tested for in the table
* @return TBool true if active and false if not enabled
*/		
TBool CSimPhone::ServiceEnabled(RMobilePhone::TMobilePhoneServiceTableV1* aTable, 
								TInt aOffset, 
								TUint8 aModifier)
	{
	if (aTable)
		{
		TUint8* pService = &(aTable->iServices1To8);
		TUint8 service = *(pService+aOffset);		
		if (service & aModifier)
			{
			return true;
			}
		}
	return false;
	}

/**
* An IPC support function for counting the number of entries in the Acces Point Name (APN) List
*
* @param aReqHandle handle to the IPC call used for completing the round trip
* @param aIndex a pointer to some memory for storing the number of entries
* @return TInt always returns KErrNone
*/
TInt CSimPhone::EnumerateAPNEntries(TTsyReqHandle aReqHandle, TUint32* aIndex)
	{
	if (IsICCLocked())
		{
		SYMBIAN_REQEXTERR(aReqHandle, KErrGsm0707SimPin1Required, KErrAccessDenied);
		return KErrNone;
		}
		
	if (ServiceEnabled(iUSIMServiceTable,iAPNServiceOffset,RMobilePhone::KUstACL))
		{
		if (ServiceEnabled(iUSIMEnabledServiceTable,iAPNServiceOffset,RMobilePhone::KUstACL))
			{
			*aIndex = iAPNList.Count();
			ReqCompleted(aReqHandle,KErrNone);
			}
		else
			{
			SYMBIAN_REQEXTERR(aReqHandle,KErrMMEtelAPNEnabledServicesTableNotFound,KErrNotFound);
			}		
		}
	else
		{
		SYMBIAN_REQEXTERR(aReqHandle,KErrMMEtelAPNNameACLNotFound,KErrNotSupported);
		}

	return KErrNone;
	}
	
/**
* An IPC support function for retrieving a name in the Acces Point Name (APN) List
*
* @param aReqHandle handle to the IPC call used for completing the round trip
* @param aIndex a index into the APN List for the name to be retrieved
* @param aRetrieved a TAPNEntryV3Pckg to store the name from the APN List
* @return TInt always returns KErrNone
*/	
TInt CSimPhone::GetAPNname(	TTsyReqHandle aReqHandle, 
							TUint32& aIndex, 
							RMobilePhone::TAPNEntryV3Pckg& aRetrieved)
	{
	if (IsICCLocked())
		{
		ReqCompleted(aReqHandle,SYMBIAN_EXTERR(	KErrGsm0707SimPin1Required,	KErrAccessDenied));		
		return KErrNone;
		}
			
	if (ServiceEnabled(iUSIMServiceTable,iAPNServiceOffset,RMobilePhone::KUstACL))
		{
		if (ServiceEnabled(iUSIMEnabledServiceTable,iAPNServiceOffset,RMobilePhone::KUstACL))
			{	
			if (aIndex < iAPNList.Count() )
				{
				RMobilePhone::TAPNEntryV3& tAPNEntryV3 = aRetrieved();
				
				tAPNEntryV3.iApn.Zero();
				tAPNEntryV3.iApn.Insert(0,iAPNList[aIndex].iApn);
				
				ReqCompleted(aReqHandle,KErrNone);
				}
			else
				{
				ReqCompleted(aReqHandle,SYMBIAN_EXTERR(KErrMMEtelAPNNameInvalidIndex,KErrNotFound));
				}
			}
		else
			{
			ReqCompleted(aReqHandle,SYMBIAN_EXTERR(	KErrMMEtelAPNEnabledServicesTableNotFound,
													KErrNotFound));
			}		
		}
	else
		{
		ReqCompleted(aReqHandle,SYMBIAN_EXTERR(	KErrMMEtelAPNNameACLNotFound,
												KErrNotSupported));
		}
	return KErrNone;		
	}
	
/**
* An IPC support function for adding a name to the Acces Point Name (APN) List
*
* @param aReqHandle handle to the IPC call used for completing the round trip
* @param aIndex a index into the APN List for the name to be retrieved
* @param aEntry a TAPNEntryV3Pckg to add to the end of the APN List
* @return TInt always returns KErrNone
*/				
TInt CSimPhone::AppendAPNName(TTsyReqHandle aReqHandle, RMobilePhone::TAPNEntryV3Pckg& aEntry)
	{
	if (IsPIN2Locked())
		{
		ReqCompleted(aReqHandle,SYMBIAN_EXTERR(	KErrGsm0707SimPin2Required,	KErrAccessDenied));		
		return KErrNone;
		}
			
	if (ServiceEnabled(iUSIMServiceTable,iAPNServiceOffset,RMobilePhone::KUstACL))
		{
		if (ServiceEnabled(iUSIMEnabledServiceTable,iAPNServiceOffset,RMobilePhone::KUstACL))
			{		
			RMobilePhone::TAPNEntryV3& entry = (aEntry)();
			iAPNList.Append(entry);
			
			ReqCompleted(aReqHandle,KErrNone);
			
			if (iAPNListNotify.iNotifyPending == TRUE)
				{
				iAPNListNotify.iNotifyPending = EFalse;
				ReqCompleted(iAPNListNotify.iNotifyHandle,KErrNone);
				}
			}
		else
			{
			ReqCompleted(aReqHandle,SYMBIAN_EXTERR(	KErrMMEtelAPNEnabledServicesTableNotFound,
													KErrNotFound));
			}		
		}
	else
		{
		ReqCompleted(aReqHandle,SYMBIAN_EXTERR(	KErrMMEtelAPNNameACLNotFound,
												KErrNotSupported));
		}
			
	return KErrNone;
	}	
	
/**
* An IPC support function for removing a name from the Acces Point Name (APN) List
*
* @param aReqHandle handle to the IPC call used for completing the round trip
* @param aIndex a index into the APN List for the name to be deleted
* @return TInt always returns KErrNone
*/		
TInt CSimPhone::DeleteAPNName(TTsyReqHandle aReqHandle, TUint32& aIndex)
	{
	if (IsPIN2Locked())
		{
		ReqCompleted(aReqHandle,SYMBIAN_EXTERR(	KErrGsm0707SimPin2Required,	KErrAccessDenied));		
		return KErrNone;
		}
			
	if (ServiceEnabled(iUSIMServiceTable,iAPNServiceOffset,RMobilePhone::KUstACL))
		{
		if (ServiceEnabled(iUSIMEnabledServiceTable,iAPNServiceOffset,RMobilePhone::KUstACL))
			{		
			if (aIndex < iAPNList.Count())
				{
				iAPNList.Remove(aIndex);
				iAPNList.Compress();
				ReqCompleted(aReqHandle,KErrNone);
				if (iAPNListNotify.iNotifyPending == TRUE)
					{
					iAPNListNotify.iNotifyPending = EFalse;
					ReqCompleted(iAPNListNotify.iNotifyHandle,KErrNone);
					}
				}
			else
				{
				ReqCompleted(aReqHandle,SYMBIAN_EXTERR(KErrMMEtelAPNNameInvalidIndex,KErrNotFound));			
				}
			}
		else
			{
			ReqCompleted(aReqHandle,SYMBIAN_EXTERR(	KErrMMEtelAPNEnabledServicesTableNotFound,
													KErrNotFound));
			}		
		}
	else
		{
		ReqCompleted(aReqHandle,SYMBIAN_EXTERR(	KErrMMEtelAPNNameACLNotFound,
												KErrNotSupported));
		}
	return KErrNone;
	}

/**
* An IPC support function for registering interest in APN List changes
*
* @param aReqHandle handle to the IPC call used for completing the round trip
* @return TInt always returns KErrNone
*/		
TInt CSimPhone::NotifyAPNListChanged(TTsyReqHandle aReqHandle)
	{
	if (ServiceEnabled(iUSIMServiceTable,iAPNServiceOffset,RMobilePhone::KUstACL))
		{
		if (ServiceEnabled(iUSIMEnabledServiceTable,iAPNServiceOffset,RMobilePhone::KUstACL))
			{
			iAPNListNotify.iNotifyPending = ETrue;
			iAPNListNotify.iNotifyHandle = aReqHandle;
			}
		else
			{
			ReqCompleted(aReqHandle,SYMBIAN_EXTERR(	KErrMMEtelAPNEnabledServicesTableNotFound,
													KErrNotFound));
			}		
		}
	else
		{
		ReqCompleted(aReqHandle,SYMBIAN_EXTERR(	KErrMMEtelAPNNameACLNotFound,
												KErrNotSupported));
		}
	return KErrNone;	
	}
	
/**
* An IPC support function for switching the Acces Point Name (APN) List to enabled or disabled
*
* @param aReqHandle handle to the IPC call used for completing the round trip
* @param aAPNControlListServiceStatus service status to be set
* @return TInt always returns KErrNone
*/		
TInt CSimPhone::SetAPNControlListServiceStatus(TTsyReqHandle aReqHandle, 
		RMobilePhone::TAPNControlListServiceStatus& aAPNControlListServiceStatus)	
	{
	if (IsPIN2Locked())
		{
		ReqCompleted(aReqHandle,SYMBIAN_EXTERR(	KErrGsm0707SimPin2Required,	KErrAccessDenied));		
		return KErrNone;
		}
			
	if (ServiceEnabled(iUSIMServiceTable,iAPNServiceOffset,RMobilePhone::KUstACL))
		{
		switch (aAPNControlListServiceStatus)
			{
			case RMobilePhone::EAPNControlListServiceDisabled:
				{
				if (iUSIMEnabledServiceTable)
					{
					iUSIMEnabledServiceTable->iServices33To40 = 
					(iUSIMEnabledServiceTable->iServices33To40 & RMobilePhone::KUstACL) 
						^ iUSIMEnabledServiceTable->iServices33To40;
						
					ReqCompleted(aReqHandle,KErrNone);
					if (iAPNStatusNotify.iNotifyPending == TRUE)
						{
						*static_cast<RMobilePhone::TAPNControlListServiceStatus*>(iAPNStatusNotify.iNotifyData) = 
								RMobilePhone::EAPNControlListServiceDisabled;
						iAPNStatusNotify.iNotifyPending = EFalse;
						ReqCompleted(iAPNStatusNotify.iNotifyHandle,KErrNone);
						}					
					}
				else
					{
					ReqCompleted(aReqHandle,SYMBIAN_EXTERR(	KErrMMEtelAPNEnabledServicesTableNotFound,
															KErrNotSupported));					
					}
				}
				break;				
			case RMobilePhone::EAPNControlListServiceEnabled:
				{
				if (iUSIMEnabledServiceTable)
					{
					iUSIMEnabledServiceTable->iServices33To40 |= RMobilePhone::KUstACL;
					ReqCompleted(aReqHandle,KErrNone);
					if (iAPNStatusNotify.iNotifyPending == TRUE)
						{
						*static_cast<RMobilePhone::TAPNControlListServiceStatus*>(iAPNStatusNotify.iNotifyData) = 
								RMobilePhone::EAPNControlListServiceEnabled;						
						iAPNStatusNotify.iNotifyPending = EFalse;
						ReqCompleted(iAPNStatusNotify.iNotifyHandle,KErrNone);
						}
					}
				else
					{
					ReqCompleted(aReqHandle,SYMBIAN_EXTERR(	KErrMMEtelAPNEnabledServicesTableNotFound,
															KErrNotSupported));	
					}						
				}
				break;	
			default:
				ReqCompleted(aReqHandle,SYMBIAN_EXTERR(	KErrMMEtelFormatNotSupported,
														KErrNotFound));				
			}
		}
	else
		{
		ReqCompleted(aReqHandle,SYMBIAN_EXTERR(	KErrMMEtelAPNEnabledServicesTableNotFound,
												KErrNotSupported));		
		}
	return KErrNone;	
	}
	
/**
* An IPC support function for retrieving the Acces Point Name (APN) List enabled/ disabled
* status
*
* @param aReqHandle handle to the IPC call used for completing the round trip
* @param aAPNControlListServiceStatus memory for storing the current status for the user
* @return TInt always returns KErrNone
*/		
TInt CSimPhone::GetAPNControlListServiceStatus(TTsyReqHandle aReqHandle, 
		RMobilePhone::TAPNControlListServiceStatus& aAPNControlListServiceStatus)
	{
	if (IsICCLocked())
		{
		ReqCompleted(aReqHandle,SYMBIAN_EXTERR(	KErrGsm0707SimPin1Required,	KErrAccessDenied));		
		return KErrNone;
		}
			
	if (ServiceEnabled(iUSIMServiceTable,iAPNServiceOffset,RMobilePhone::KUstACL))
		{
		if (ServiceEnabled(iUSIMEnabledServiceTable,iAPNServiceOffset,RMobilePhone::KUstACL))
			{
			aAPNControlListServiceStatus = RMobilePhone::EAPNControlListServiceEnabled;
			}
		else
			{
			aAPNControlListServiceStatus = RMobilePhone::EAPNControlListServiceDisabled;
			}	
		ReqCompleted(aReqHandle,KErrNone);	
		}
	else
		{
		ReqCompleted(aReqHandle,SYMBIAN_EXTERR(	KErrMMEtelAPNEnabledServicesTableNotFound,
												KErrNotSupported));				
		}
	return KErrNone;	
	}
	
/**
* An IPC support function for registering interest in the Acces Point Name (APN) List 
* enabled/ disabled status changing
*
* @param aReqHandle handle to the IPC call used for completing the round trip
* @param aAPNControlListServiceStatus memory for storing the current status for the user
* @return TInt always returns KErrNone
*/			
TInt CSimPhone::NotifyAPNControlListServiceStatusChange(TTsyReqHandle aReqHandle, 
		RMobilePhone::TAPNControlListServiceStatus& aAPNControlListServiceStatus)
	{
	iAPNStatusNotify.iNotifyPending = ETrue;
	iAPNStatusNotify.iNotifyHandle = aReqHandle;
	iAPNStatusNotify.iNotifyData = &aAPNControlListServiceStatus;
	return KErrNone;	
	}
	
/**
* A utility function for checking table availabilty and performing a copy on it
*
* @param aFrom The TMobilePhoneServiceTableV1 table to copied out of
* @param aTo The TMobilePhoneServiceTableV1 table to copied into
* @return TInt returns KErrNone for success and KErrNotSupported if the table aren't 
* valid
*/		
TInt CSimPhone::CopyServiceTable(RMobilePhone::TMobilePhoneServiceTableV1* aFrom, 
		RMobilePhone::TMobilePhoneServiceTableV1* aTo)
	{
	if (aFrom && aTo)
		{
		*aTo = *aFrom;
		return KErrNone;
		}
	else 
		{
		// This value is returned because this function tests the existance of the table 
		// not just that the args are OK.
		return SYMBIAN_EXTERR(KErrMMEtelAPNEnabledServicesTableNotFound, KErrNotSupported);
		}
	}
	
/**
 * A utility function for checking table availabilty and performing a copy on it
 *
 * @param aFrom The TMobilePhoneServiceTableV8 table to copied out of
 * @param aTo The TMobilePhoneServiceTableV8 table to copied into
 * @return TInt returns KErrNone for success and KErrNotSupported if the table aren't 
 * valid
 */		
 TInt CSimPhone::CopyServiceTableV8(RMobilePhone::TMobilePhoneServiceTableV8* aFrom, 
 		RMobilePhone::TMobilePhoneServiceTableV8* aTo)
 	{
 	if (aFrom && aTo)
 		{
 		*aTo = *aFrom;
 		return KErrNone;
 		}
 	else 
 		{
 		// This value is returned because this function tests the existance of the table 
 		// not just that the args are OK.
 		return SYMBIAN_EXTERR(KErrMMEtelAPNEnabledServicesTableNotFound, KErrNotSupported);
 		}
 	}

/**
* An IPC support function for retrieving one of the service tables
*
* @param aReqHandle handle to the IPC call used for completing the round trip
* @param aTable the type of table the user has interest in
* @param aPCmd memory to store the retrieved table
* @return TInt always returns KErrNone
*/		
TInt CSimPhone::GetServiceTable(TTsyReqHandle aReqHandle, 
		RMobilePhone::TMobilePhoneServiceTable aTable, 
		TDes8* aPCmd)	
	{
	TInt ret = KErrNone;
	RMobilePhone::TMobilePhoneServiceTableV1Pckg *pTablePckg = (RMobilePhone::TMobilePhoneServiceTableV1Pckg*) aPCmd;
	RMobilePhone::TMobilePhoneServiceTableV1 &aSst = (*pTablePckg)();
	
	switch (aTable)
		{

		/** Retrieve SIM service table from active SIM application on UICC.
		 Modes: GSM/WCDMA */
		case RMobilePhone::ESIMServiceTable:
			ret = CopyServiceTable (iSIMServiceTable, &aSst);
			break;
			/** Retrieve USIM service table from active USIM application on UICC.
			 
			 Modes: WCDMA */
		case RMobilePhone::EUSIMServiceTable:
			if ( aSst.ExtensionId ()== KEtelExtMultimodeV8)
				{
				RMobilePhone::TMobilePhoneServiceTableV8Pckg* pTableV8Pckg = (RMobilePhone::TMobilePhoneServiceTableV8Pckg*) aPCmd;
				RMobilePhone::TMobilePhoneServiceTableV8 &pTableV8 = (*pTableV8Pckg) ();
				ret = CopyServiceTableV8 (iUSIMServiceTableV8, &pTableV8);
				}
			else
				{
				ret = CopyServiceTable (iUSIMServiceTable, &aSst);
				}
			break;
			/** USIM Enabled Services Table to be used in conjunction with the USIM Service Table.
			 
			 Modes: WCDMA */
		case RMobilePhone::EUSIMEnabledServiceTable:
			ret = CopyServiceTable (iUSIMEnabledServiceTable, &aSst);
			break;
		default:
			ReqCompleted (aReqHandle, SYMBIAN_EXTERR( KErrMMEtelFormatNotSupported,
					KErrArgument));
			return KErrNone;
		}
	// Extended error codes are handled here by the CopyServiceTable() func.
	ReqCompleted(aReqHandle,ret);
	return KErrNone;
	}

TInt CSimPhone::GetPhoneId(TTsyReqHandle aReqHandle,RMobilePhone::TMobilePhoneIdentityV1* aPhoneId)
	{

	//complete with error if error specified.
	if(iPhoneId.iError != KErrNone)
		{
		ReqCompleted(aReqHandle,iPhoneId.iError);
		return KErrNone;
		}
	//fill out phone identity.
	if(iPhoneIdCaps & RMobilePhone:: KCapsGetManufacturer)
		{
		if(iPhoneId.iManufacturerId.Length() > RMobilePhone::KPhoneManufacturerIdSize)
			aPhoneId->iManufacturer.Copy(iPhoneId.iManufacturerId.Left(RMobilePhone::KPhoneManufacturerIdSize));
		else
			aPhoneId->iManufacturer.Copy(iPhoneId.iManufacturerId);
		}

	if(iPhoneIdCaps & RMobilePhone::KCapsGetModel)
		{
		if(iPhoneId.iModelId.Length() > RMobilePhone::KPhoneModelIdSize)
			aPhoneId->iModel.Copy(iPhoneId.iModelId.Left(RMobilePhone::KPhoneModelIdSize));
		else
			aPhoneId->iModel.Copy(iPhoneId.iModelId);
		}

	if(iPhoneIdCaps & RMobilePhone::KCapsGetRevision)
		{
		if(iPhoneId.iRevisionId.Length() > RMobilePhone::KPhoneRevisionIdSize)
			aPhoneId->iRevision.Copy(iPhoneId.iRevisionId.Left(RMobilePhone::KPhoneRevisionIdSize));
		else
			aPhoneId->iRevision.Copy(iPhoneId.iRevisionId);
		}

	if(iPhoneIdCaps & RMobilePhone::KCapsGetSerialNumber)
		{
		if(iPhoneId.iSerialId.Length() > RMobilePhone::KPhoneSerialNumberSize)
			aPhoneId->iSerialNumber.Copy(iPhoneId.iSerialId.Left(RMobilePhone::KPhoneSerialNumberSize));
		else
			aPhoneId->iSerialNumber.Copy(iPhoneId.iSerialId);
		}

	ReqCompleted(aReqHandle,KErrNone);
	return KErrNone;
	}

/**
 * Callback function invoked by the observer object when test number property is changed.
 * This function is supposed to reset SIMTSY. Currently only SMS messaging part of SIMTSY
 * is re-started with the new test number.
 */
void CSimPhone::HandleTestNumberChangedL()
    {
    SetTestNumberAndReadConfigurationFileL();
    iSmsMessaging->ReloadConfigurationSettingsL();
    }

TInt CSimPhone::CheckConfigFile()
    {
    TInt testNumber;
    User::LeaveIfError(GetTestNumber(testNumber));
    iSectionName.Format(KSectionNameFormat,testNumber);
    
    CTestConfig* configFile = NULL; 

    TRAPD(err, configFile = CTestConfig::NewLC(iFs, KConfigFileDir, KConfigFilename); CleanupStack::Pop(configFile));
    if( err != KErrNone || configFile->Section(iSectionName) != NULL )
        {
        err = KErrNone;
        }
    else
        {
        err = KErrNotFound;
        }
    delete configFile;
    return err;
    }
/**
	Constructor for suspend call processing timer
*/
CSimPhone::CSetCallProcessingSuspendStateTimerCallBack::CSetCallProcessingSuspendStateTimerCallBack()
	{}

/**
	Must be called before setting the timer. Simply stores the request to be completed when
	timer expires

 @param the pointer to the phone that the request will be completed on when the timer expires
 @param aReqHandle handle to request that will be completed when the timer expires
*/
void CSimPhone::CSetCallProcessingSuspendStateTimerCallBack::SetHandle(CSimPhone* aPhone, const TTsyReqHandle aHandle)
	{
	iPhone = aPhone;
	iHandle = aHandle;
	}

/**
	Implementation of MTimerCallback::TimerCallBack
	Simply completes the suspend call processing request after the delay
*/
void CSimPhone::CSetCallProcessingSuspendStateTimerCallBack::TimerCallBack(TInt /*aId*/)
	{
	iPhone->ReqCompleted(iHandle,KErrNone);
	}

/**
	Constructor for network mode processing timer
*/
CSimPhone::CNetworkModeTimerCallBack::CNetworkModeTimerCallBack()
	{
	}

void CSimPhone::CNetworkModeTimerCallBack::SetHandle(CSimPhone* aPhone)
	{
	iPhone = aPhone;
	}
/**
	Implementation of MTimerCallback::TimerCallBack
	Simply completes the suspend call processing request after the delay
*/
void CSimPhone::CNetworkModeTimerCallBack::TimerCallBack(TInt /*aId*/)
	{
	LOGPACKET1(">>CSimPhone::CNetworkModeTimerCallBack::TimerCallBack");
	iPhone->TimerCallBackNetworkMode();
	}

TInt CSimPhone::NotifyModeChange(const TTsyReqHandle aTsyReqHandle, RMobilePhone::TMobilePhoneNetworkMode* aCaps)
	{
	LOGPACKET1("CSimPhone::NotifyModeChange called");
	__ASSERT_ALWAYS(!iNotifyNetworkModeChange.iNotifyPending,SimPanic(ENotificationAlreadyPending));
 
	if (iNetworkModeArray->Count() == 0)
		{
		ReqCompleted(aTsyReqHandle,KErrNotSupported);
		return KErrNone;
		}
	
	iNotifyNetworkModeChange.iNotifyPending = ETrue;
	iNotifyNetworkModeChange.iNotifyHandle = aTsyReqHandle;
	iNotifyNetworkModeChange.iNotifyData = aCaps;
	return KErrNone;
	}

/**
* Cancels a clients interest in the change of dynamic caps.
*
* @param aTsyReqHandle Tsy Request handle for the client cancel request
* @return KErrNone
*/
TInt CSimPhone::NotifyModeChangeCancel(const TTsyReqHandle aTsyReqHandle)
	{
	LOGPACKET1("CSimPhone::NotifyModeChangeCancel called");
	if( (iNotifyNetworkModeChange.iNotifyPending) && (aTsyReqHandle == iNotifyNetworkModeChange.iNotifyHandle))
		{
		iNotifyNetworkModeChange.iNotifyPending=EFalse;
		ReqCompleted(aTsyReqHandle,KErrCancel);
		}
	return KErrNone;
	}

/**
* Timer callback function to simulate a change in the Network Mode of the phone
*/
void CSimPhone::TimerCallBackNetworkMode()
	{
	if (++iNetworkModeIndex >= iNetworkModeArray->Count())
		{
		return;
		}
	iNtwkMode = iNetworkModeArray->At(iNetworkModeIndex).iNetworkMode;
	if (iNotifyNetworkModeChange.iNotifyPending)
		{
		iNotifyNetworkModeChange.iNotifyPending = EFalse;
		*(RMobilePhone::TMobilePhoneNetworkMode*)iNotifyNetworkModeChange.iNotifyData = (RMobilePhone::TMobilePhoneNetworkMode)iNtwkMode;
		ReqCompleted(iNotifyNetworkModeChange.iNotifyHandle,KErrNone);
		}
	
	iNetworkModeTimer->Start(iNetworkModeArray->At(iNetworkModeIndex).iDuration, &iTimerCallBackNetworkMode);
	}

TInt CSimPhone::CheckSimTsyVersion(RMobilePhone::TMultimodeType& aDataStruct) // overload this for other types of structures
/**
* Checks the version of a data structure against the (simulated) version of SIMTSY, in order
* to determine if the version of the API used is compatible with the version of the TSY used.
*
* @param aDataStruct A versioned data-holding structure, derived from RMobilePhone::TMultimodeType
*
* @return KErrNone if the version of aDataStruct is supported by the user-defined simulated TSY version. KErrNotSupported otherwise
*/
	{
	TInt ret = KErrNone;
	
	//retrieval of data version
	TInt dataVersion;
	switch (aDataStruct.ExtensionId())
		{
		case KETelExtMultimodeV1:
		case RMobilePhoneStore::KETelMobilePhonebookStoreV1:
		case RMobileSmsMessaging::KETelMobileSmsSendAttributesV1:
		case KETelExt3rdPartyV1:
			dataVersion = 1;
			break;
		case KETelExtMultimodeV2:
		//case KETelMobilePhonebookStoreV2: - omitted because it is equal to KETelExtMultimodeV2
			dataVersion = 2;
			break;
		case KETelExtMultimodeV3:
			dataVersion = 3;
			break;
		case KETelExtMultimodeV4:
			dataVersion = 4;
			break;
		case KEtelExtMultimodeV5:
		//case KETelMobilePhonebookStoreV5: - omitted because it is equal to KETelExtMultimodeV5
			dataVersion = 5;
			break;
		case KEtelExtMultimodeV6:
			dataVersion = 6;
			break;
		case KEtelExtMultimodeV7:
			dataVersion = 7;
			break;
		case KEtelExtMultimodeV8:
			dataVersion = 8;
			break;
		case KEtelExtMultimodeV9:
			dataVersion = 9;
			break;
		default:
			dataVersion = KSimTsyDefaultVersion; 
			break;
		}
		
	if(iSimTsyVersion < dataVersion)
		{
		ret = KErrNotSupported;
		}
		
	return ret;	
	}

TInt CSimPhone::CheckSimTsyVersion(TPacketBase& aDataStruct)
/**
* Checks the version of a data structure against the (simulated) version of SIMTSY, in order
* to determine if the version of the API used is compatible with the version of the TSY used.
*
* @param aDataStruct A versioned data-holding structure, derived from TPacketBase
*
* @return KErrNone if the version of aDataStruct is supported by the user-defined simulated TSY version. KErrNotSupported otherwise
*/
	{
	TInt ret = KErrNone;
	
	//retrieval of data version
	TInt dataVersion;
	switch (aDataStruct.ExtensionId())
		{
		case KETelExtPcktV1:
			dataVersion = 1;
			break;
		case KETelExtPcktV2:
			dataVersion = 2;
			break;
		case KETelExtPcktV3:
			dataVersion = 3;
			break;
		default:
			dataVersion = KSimTsyDefaultVersion; 
			break;
		}
		
	if(iSimTsyVersion < dataVersion)
		{
		ret = KErrNotSupported;
		}
		
	return ret;	
	}


TInt CSimPhone::CheckSimTsyVersion(RCall::TCallParams& aDataStruct)
/**
* Checks the version of a data structure against the (simulated) version of SIMTSY, in order
* to determine if the version of the API used is compatible with the version of the TSY used.
*
* @param aDataStruct A versioned data-holding structure, derived from TCallParams
*
* @return KErrNone if the version of aDataStruct is supported by the user-defined simulated TSY version. KErrNotSupported otherwise
*/
	{
	TInt ret = KErrNone;

	//retrieval of data version
	TInt dataVersion;
	switch (aDataStruct.ExtensionId())
		{
		case RMobileCall::KETelMobileCallParamsV1:
		case RMobileCall::KETelMobileDataCallParamsV1:
		case RMobileCall::KETelMobileHscsdCallParamsV1:
		case RMobileCall::KETel3rdPartyCallParamsV1:
			dataVersion = 1;
			break;
		case RMobileCall::KETelMobileCallParamsV2:
		case RMobileCall::KETelMobileDataCallParamsV2:
		case RMobileCall::KETelMobileHscsdCallParamsV2:
			dataVersion = 2;
			break;
		default:
			dataVersion = KSimTsyDefaultVersion; 
			break;
		}
		
	if(iSimTsyVersion < dataVersion)
		{
		ret = KErrNotSupported;
		}
		
	return ret;	
	}


TInt CSimPhone::CheckSimTsyVersion(RMobilePhone::CImsAuthorizationInfoV5& /*aDataStruct*/) // overload this for other types of structures
/**
* Checks the version of a data structure against the (simulated) version of SIMTSY, in order
* to determine if the version of the API used is compatible with the version of the TSY used.
*
* @param aDataStruct A versioned data-holding structure, of type RMobilePhone::CImsAuthorizationInfoV5 or a class derived from it.
*
* @return KErrNone if the version of aDataStruct is supported by the user-defined simulated TSY version. KErrNotSupported otherwise
*/
	{

	TInt ret = KErrNone;
	
	//retrieval of data version
	TInt dataVersion = 5; // (does not derive from the TMultimodeType data structure as usual...)
		
	if(iSimTsyVersion < dataVersion)
		{
		ret = KErrNotSupported;
		}
		
	return ret;	
	}
	

TInt CSimPhone::CheckSimTsyVersion(RPacketContext::CTFTMediaAuthorizationV3& /*aDataStruct*/) // overload this for other types of structures
/**
* Checks the version of a data structure against the (simulated) version of SIMTSY, in order
* to determine if the version of the API used is compatible with the version of the TSY used.
*
* @param aDataStruct A versioned data-holding structure, of type RPacketContext::CTFTMediaAuthorizationV3 or a class derived from it.
*
* @return KErrNone if the version of aDataStruct is supported by the user-defined simulated TSY version. KErrNotSupported otherwise
*/
	{

	TInt ret = KErrNone;
	
	//retrieval of data version
	TInt dataVersion = 3; // (does not derive from the TMultimodeType data structure as usual...)
		
	if(iSimTsyVersion < dataVersion)
		{
		ret = KErrNotSupported;
		}
		
	return ret;	
	}
////////////////////
// CSimTestNumberObserver
////////////////////

CSimPhone::CSimTestNumberObserver::CSimTestNumberObserver(CSimPhone& aSimPhone)
: CActive(CActive::EPriorityStandard),
  iSimPhone(aSimPhone)
    {
    CActiveScheduler::Add(this);
    }

CSimPhone::CSimTestNumberObserver::~CSimTestNumberObserver()
    {
    Cancel();
    iProperty.Close();
    }

CSimPhone::CSimTestNumberObserver* CSimPhone::CSimTestNumberObserver::NewL(CSimPhone& aSimPhone)
    {
    CSimPhone::CSimTestNumberObserver* self = new(ELeave)CSimPhone::CSimTestNumberObserver(aSimPhone);
    CleanupStack::PushL(self);
    self->ConstructL();
    CleanupStack::Pop(self);
    return self;
    }

void CSimPhone::CSimTestNumberObserver::ConstructL()
    {
    LOGPHONE1("CSimPhone::CSimPhoneObserver::ConstructL");
    User::LeaveIfError(iProperty.Attach(KUidPSSimTsyCategory, KPSSimTsyTestNumber));
    Start();
    }

void CSimPhone::CSimTestNumberObserver::Start()
    {
    LOGPHONE1("CSimPhone::CSimPhoneObserver::Start");
    iProperty.Subscribe(iStatus);
    SetActive();
    }

void CSimPhone::CSimTestNumberObserver::RunL()
    {
    LOGPHONE2("CSimPhone::CSimPhoneObserver::RunL [iStatus=%d]", iStatus.Int());
    TInt err = iSimPhone.CheckConfigFile();
    if( err == KErrNone )
        {
        iSimPhone.HandleTestNumberChangedL();
        }

    Start();
    }

void CSimPhone::CSimTestNumberObserver::DoCancel()
    {
    LOGPHONE1("CSimPhone::CSimPhoneObserver::DoCancel");
    iProperty.Cancel();
    }